home *** CD-ROM | disk | FTP | other *** search
/ Chip 2004 December / 2004-12 CHIP.iso / Internet / NVU 0.50 for Windows / nvu-0.50-win32-installer-full.exe / {app} / chrome / comm.jar / content / editor / ComposerCommands.js < prev    next >
Encoding:
JavaScript  |  2004-09-24  |  127.1 KB  |  4,247 lines

  1. /* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
  2.  
  3. /* ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is Mozilla.org.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998-2004
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Simon Fraser (sfraser@netscape.com)
  25.  *   Ryan Cassin (rcassin@supernova.org)
  26.  *   Kathleen Brade (brade@netscape.com)
  27.  *   Daniel Glazman (glazman@netscape.com)
  28.  *   Daniel Glazman (glazman@disruptive-innovations.com), on behalf of Lindows.com
  29.  *
  30.  * Alternatively, the contents of this file may be used under the terms of
  31.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  32.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  33.  * in which case the provisions of the GPL or the LGPL are applicable instead
  34.  * of those above. If you wish to allow use of your version of this file only
  35.  * under the terms of either the GPL or the LGPL, and not to allow others to
  36.  * use your version of this file under the terms of the MPL, indicate your
  37.  * decision by deleting the provisions above and replace them with the notice
  38.  * and other provisions required by the GPL or the LGPL. If you do not delete
  39.  * the provisions above, a recipient may use your version of this file under
  40.  * the terms of any one of the MPL, the GPL or the LGPL.
  41.  *
  42.  * ***** END LICENSE BLOCK ***** */
  43.  
  44.  
  45. /* Implementations of nsIControllerCommand for composer commands */
  46.  
  47. var gComposerJSCommandControllerID = 0;
  48.  
  49.  
  50. //-----------------------------------------------------------------------------------
  51. function SetupHTMLEditorCommands()
  52. {
  53.   var commandTable = GetComposerCommandTable();
  54.   if (!commandTable)
  55.     return;
  56.   
  57.   // Include everthing a text editor does
  58.   SetupTextEditorCommands();
  59.  
  60.   //dump("Registering HTML editor commands\n");
  61.  
  62.   commandTable.registerCommand("cmd_renderedHTMLEnabler", nsDummyHTMLCommand);
  63.  
  64.   commandTable.registerCommand("cmd_grid",  nsGridCommand);
  65.  
  66.   commandTable.registerCommand("cmd_listProperties",  nsListPropertiesCommand);
  67.   commandTable.registerCommand("cmd_pageProperties",  nsPagePropertiesCommand);
  68.   commandTable.registerCommand("cmd_colorProperties", nsColorPropertiesCommand);
  69.   commandTable.registerCommand("cmd_advancedProperties", nsAdvancedPropertiesCommand);
  70.   commandTable.registerCommand("cmd_objectProperties",   nsObjectPropertiesCommand);
  71.   commandTable.registerCommand("cmd_removeNamedAnchors", nsRemoveNamedAnchorsCommand);
  72.   commandTable.registerCommand("cmd_editLink",        nsEditLinkCommand);
  73.   
  74.   commandTable.registerCommand("cmd_form",          nsFormCommand);
  75.   commandTable.registerCommand("cmd_inputtag",      nsInputTagCommand);
  76.   commandTable.registerCommand("cmd_inputimage",    nsInputImageCommand);
  77.   commandTable.registerCommand("cmd_textarea",      nsTextAreaCommand);
  78.   commandTable.registerCommand("cmd_select",        nsSelectCommand);
  79.   commandTable.registerCommand("cmd_button",        nsButtonCommand);
  80.   commandTable.registerCommand("cmd_label",         nsLabelCommand);
  81.   commandTable.registerCommand("cmd_fieldset",      nsFieldSetCommand);
  82.   commandTable.registerCommand("cmd_isindex",       nsIsIndexCommand);
  83.   commandTable.registerCommand("cmd_image",         nsImageCommand);
  84.   commandTable.registerCommand("cmd_hline",         nsHLineCommand);
  85.   commandTable.registerCommand("cmd_link",          nsLinkCommand);
  86.   commandTable.registerCommand("cmd_anchor",        nsAnchorCommand);
  87.   commandTable.registerCommand("cmd_insertHTMLWithDialog", nsInsertHTMLWithDialogCommand);
  88.   commandTable.registerCommand("cmd_insertBreak",   nsInsertBreakCommand);
  89.   commandTable.registerCommand("cmd_insertBreakAll",nsInsertBreakAllCommand);
  90.  
  91.   commandTable.registerCommand("cmd_table",              nsInsertOrEditTableCommand);
  92.   commandTable.registerCommand("cmd_editTable",          nsEditTableCommand);
  93.   commandTable.registerCommand("cmd_SelectTable",        nsSelectTableCommand);
  94.   commandTable.registerCommand("cmd_SelectRow",          nsSelectTableRowCommand);
  95.   commandTable.registerCommand("cmd_SelectColumn",       nsSelectTableColumnCommand);
  96.   commandTable.registerCommand("cmd_SelectCell",         nsSelectTableCellCommand);
  97.   commandTable.registerCommand("cmd_SelectAllCells",     nsSelectAllTableCellsCommand);
  98.   commandTable.registerCommand("cmd_InsertTable",        nsInsertTableCommand);
  99.   commandTable.registerCommand("cmd_InsertRowAbove",     nsInsertTableRowAboveCommand);
  100.   commandTable.registerCommand("cmd_InsertRowBelow",     nsInsertTableRowBelowCommand);
  101.   commandTable.registerCommand("cmd_InsertColumnBefore", nsInsertTableColumnBeforeCommand);
  102.   commandTable.registerCommand("cmd_InsertColumnAfter",  nsInsertTableColumnAfterCommand);
  103.   commandTable.registerCommand("cmd_InsertCellBefore",   nsInsertTableCellBeforeCommand);
  104.   commandTable.registerCommand("cmd_InsertCellAfter",    nsInsertTableCellAfterCommand);
  105.   commandTable.registerCommand("cmd_DeleteTable",        nsDeleteTableCommand);
  106.   commandTable.registerCommand("cmd_DeleteRow",          nsDeleteTableRowCommand);
  107.   commandTable.registerCommand("cmd_DeleteColumn",       nsDeleteTableColumnCommand);
  108.   commandTable.registerCommand("cmd_DeleteCell",         nsDeleteTableCellCommand);
  109.   commandTable.registerCommand("cmd_DeleteCellContents", nsDeleteTableCellContentsCommand);
  110.   commandTable.registerCommand("cmd_JoinTableCells",     nsJoinTableCellsCommand);
  111.   commandTable.registerCommand("cmd_SplitTableCell",     nsSplitTableCellCommand);
  112.   commandTable.registerCommand("cmd_TableOrCellColor",   nsTableOrCellColorCommand);
  113.   commandTable.registerCommand("cmd_NormalizeTable",     nsNormalizeTableCommand);
  114.   commandTable.registerCommand("cmd_smiley",             nsSetSmiley);
  115.   commandTable.registerCommand("cmd_ConvertToTable",     nsConvertToTable);
  116.  
  117.   commandTable.registerCommand("cmd_blockBorders",       nsBlockBordersCommand);
  118.   commandTable.registerCommand("cmd_glues",              nsGluesCommand);
  119.  
  120. }
  121.  
  122. function SetupTextEditorCommands()
  123. {
  124.   var commandTable = GetComposerCommandTable();
  125.   if (!commandTable)
  126.     return;
  127.   
  128.   //dump("Registering plain text editor commands\n");
  129.   
  130.   commandTable.registerCommand("cmd_find",       nsFindCommand);
  131.   commandTable.registerCommand("cmd_findNext",   nsFindAgainCommand);
  132.   commandTable.registerCommand("cmd_findPrev",   nsFindAgainCommand);
  133.   commandTable.registerCommand("cmd_rewrap",     nsRewrapCommand);
  134.   commandTable.registerCommand("cmd_spelling",   nsSpellingCommand);
  135.   commandTable.registerCommand("cmd_validate",   nsValidateCommand);
  136.   commandTable.registerCommand("cmd_cleanup",    nsCleanupCommand);
  137.   commandTable.registerCommand("cmd_checkLinks", nsCheckLinksCommand);
  138.   commandTable.registerCommand("cmd_insertChars", nsInsertCharsCommand);
  139. }
  140.  
  141. function SetupComposerWindowCommands()
  142. {
  143.   // Don't need to do this if already done
  144.   if (gComposerWindowControllerID)
  145.     return;
  146.  
  147.   // Create a command controller and register commands
  148.   //   specific to Web Composer window (file-related commands, HTML Source...)
  149.   //   We can't use the composer controller created on the content window else
  150.   //     we can't process commands when in HTMLSource editor
  151.   // IMPORTANT: For each of these commands, the doCommand method 
  152.   //            must first call FinishHTMLSource() 
  153.   //            to go from HTML Source mode to any other edit mode
  154.  
  155.   var windowControllers = window.controllers;
  156.  
  157.   if (!windowControllers) return;
  158.  
  159.   var commandTable;
  160.   var composerController;
  161.   var editorController;
  162.   try {
  163.     composerController = Components.classes["@mozilla.org/embedcomp/base-command-controller;1"].createInstance();
  164.  
  165.     editorController = composerController.QueryInterface(Components.interfaces.nsIControllerContext);
  166.     editorController.init(null); // init it without passing in a command table
  167.  
  168.     // Get the nsIControllerCommandTable interface we need to register commands
  169.     var interfaceRequestor = composerController.QueryInterface(Components.interfaces.nsIInterfaceRequestor);
  170.     commandTable = interfaceRequestor.getInterface(Components.interfaces.nsIControllerCommandTable);
  171.   }
  172.   catch (e)
  173.   {
  174.     dump("Failed to create composerController\n");
  175.     return;
  176.   }
  177.  
  178.  
  179.   if (!commandTable)
  180.   {
  181.     dump("Failed to get interface for nsIControllerCommandManager\n");
  182.     return;
  183.   }
  184.  
  185.   // File-related commands
  186.   commandTable.registerCommand("cmd_open",           nsOpenCommand);
  187.   commandTable.registerCommand("cmd_openInTab",      nsOpenInTabCommand);
  188.   commandTable.registerCommand("cmd_save",           nsSaveCommand);
  189.   commandTable.registerCommand("cmd_saveAs",         nsSaveAsCommand);
  190.   commandTable.registerCommand("cmd_exportToText",   nsExportToTextCommand);
  191.   commandTable.registerCommand("cmd_saveAndChangeEncoding",  nsSaveAndChangeEncodingCommand);
  192.   commandTable.registerCommand("cmd_publish",        nsPublishCommand);
  193.   commandTable.registerCommand("cmd_publishAs",      nsPublishAsCommand);
  194.   commandTable.registerCommand("cmd_publishSettings",nsPublishSettingsCommand);
  195.   commandTable.registerCommand("cmd_revert",         nsRevertCommand);
  196.   commandTable.registerCommand("cmd_openRemote",     nsOpenRemoteCommand);
  197.   commandTable.registerCommand("cmd_preview",        nsPreviewCommand);
  198.   commandTable.registerCommand("cmd_editSendPage",   nsSendPageCommand);
  199.   commandTable.registerCommand("cmd_print",          nsPrintCommand);
  200.   commandTable.registerCommand("cmd_printSetup",     nsPrintSetupCommand);
  201.   commandTable.registerCommand("cmd_quit",           nsQuitCommand);
  202.   commandTable.registerCommand("cmd_close",          nsCloseCommand);
  203.   commandTable.registerCommand("cmd_closeAll",       nsCloseAllCommand);
  204.   commandTable.registerCommand("cmd_preferences",    nsPreferencesCommand);
  205.   commandTable.registerCommand("cmd_newEdited",      nsNewEditedObjectCommand);
  206.  
  207.   // Edit Mode commands
  208.   if (GetCurrentEditorType() == "html")
  209.   {
  210.     commandTable.registerCommand("cmd_NormalMode",         nsNormalModeCommand);
  211.     commandTable.registerCommand("cmd_AllTagsMode",        nsAllTagsModeCommand);
  212.     commandTable.registerCommand("cmd_HTMLSourceMode",     nsHTMLSourceModeCommand);
  213.     commandTable.registerCommand("cmd_PreviewMode",        nsPreviewModeCommand);
  214.     commandTable.registerCommand("cmd_FinishHTMLSource",   nsFinishHTMLSource);
  215.     commandTable.registerCommand("cmd_CancelHTMLSource",   nsCancelHTMLSource);
  216.     commandTable.registerCommand("cmd_updateStructToolbar", nsUpdateStructToolbarCommand);
  217.   }
  218.  
  219.   windowControllers.insertControllerAt(0, editorController);
  220.  
  221.   // Store the controller ID so we can be sure to get the right one later
  222.   gComposerWindowControllerID = windowControllers.getControllerId(editorController);
  223. }
  224.  
  225. //-----------------------------------------------------------------------------------
  226. function GetComposerCommandTable()
  227. {
  228.   var controller;
  229.   if (gComposerJSCommandControllerID)
  230.   {
  231.     try { 
  232.       controller = window.content.controllers.getControllerById(gComposerJSCommandControllerID);
  233.     } catch (e) {}
  234.   }
  235.   if (!controller)
  236.   {
  237.     //create it
  238.     controller = Components.classes["@mozilla.org/embedcomp/base-command-controller;1"].createInstance();
  239.  
  240.     var editorController = controller.QueryInterface(Components.interfaces.nsIControllerContext);
  241.     editorController.init(null);
  242.     editorController.setCommandContext(GetCurrentEditorElement());
  243.     window.content.controllers.insertControllerAt(0, controller);
  244.   
  245.     // Store the controller ID so we can be sure to get the right one later
  246.     gComposerJSCommandControllerID = window.content.controllers.getControllerId(controller);
  247.   }
  248.  
  249.   if (controller)
  250.   {
  251.     var interfaceRequestor = controller.QueryInterface(Components.interfaces.nsIInterfaceRequestor);
  252.     return interfaceRequestor.getInterface(Components.interfaces.nsIControllerCommandTable);
  253.   }
  254.   return null;
  255. }
  256.  
  257.  
  258. //-----------------------------------------------------------------------------------
  259. function UpdateCommandControllerContext()
  260. {
  261.   var controller;
  262.   if (gComposerJSCommandControllerID)
  263.   {
  264.     try { 
  265.       controller = window.content.controllers.getControllerById(gComposerJSCommandControllerID);
  266.       controller.setCommandContext(GetCurrentEditorElement());
  267.     } catch (e) {}
  268.   }
  269.   if (gComposerWindowControllerID)
  270.   {
  271.     try { 
  272.       controller = window.content.controllers.getControllerById(gComposerWindowControllerID);
  273.       controller.setCommandContext(GetCurrentEditorElement());
  274.     } catch (e) {}
  275.   }
  276.  
  277. }
  278.  
  279. //-----------------------------------------------------------------------------------
  280. function goUpdateCommandState(command)
  281. {
  282.   try
  283.   {
  284.     var controller = top.document.commandDispatcher.getControllerForCommand(command);
  285.     if (!(controller instanceof Components.interfaces.nsICommandController))
  286.       return;
  287.  
  288.     var params = newCommandParams();
  289.     if (!params) return;
  290.  
  291.     controller.getCommandStateWithParams(command, params);
  292.  
  293.     switch (command)
  294.     {
  295.       case "cmd_bold":
  296.       case "cmd_italic":
  297.       case "cmd_underline":
  298.       case "cmd_var":
  299.       case "cmd_samp":
  300.       case "cmd_code":
  301.       case "cmd_acronym":
  302.       case "cmd_abbr":
  303.       case "cmd_cite":
  304.       case "cmd_strong":
  305.       case "cmd_em":
  306.       case "cmd_superscript":
  307.       case "cmd_subscript":
  308.       case "cmd_strikethrough":
  309.       case "cmd_tt":
  310.       case "cmd_nobreak":
  311.       case "cmd_ul":
  312.       case "cmd_ol":
  313.       case "cmd_dt":
  314.       case "cmd_dd":
  315.         pokeStyleUI(command, params.getBooleanValue("state_all"));
  316.         break;
  317.  
  318.       case "cmd_paragraphState":
  319.       case "cmd_align":
  320.       case "cmd_highlight":
  321.       case "cmd_backgroundColor":
  322.       case "cmd_fontColor":
  323.       case "cmd_fontFace":
  324.       case "cmd_class":
  325.       case "cmd_fontSize":
  326.       case "cmd_absPos":
  327.       case "cmd_direction":
  328.         pokeMultiStateUI(command, params);
  329.         break;
  330.  
  331.       case "cmd_decreaseZIndex":
  332.       case "cmd_increaseZIndex":
  333.       case "cmd_indent":
  334.       case "cmd_outdent":
  335.       case "cmd_increaseFont":
  336.       case "cmd_decreaseFont":
  337.       case "cmd_removeStyles":
  338.       case "cmd_glues":
  339.         break;
  340.  
  341.       default: dump("no update for command: " +command+"\n");
  342.     }
  343.   }
  344.   catch (e) { dump("An error occurred updating the "+command+" command: \n"+e+"\n"); }
  345. }
  346.  
  347. function goUpdateComposerMenuItems(commandset)
  348. {
  349.   //dump("Updating commands for " + commandset.id + "\n");
  350.  
  351.   for (var i = 0; i < commandset.childNodes.length; i++)
  352.   {
  353.     var commandNode = commandset.childNodes[i];
  354.     var commandID = commandNode.id;
  355.     if (commandID)
  356.     {
  357.       goUpdateCommand(commandID);  // enable or disable
  358.       if (commandNode.hasAttribute("state"))
  359.         goUpdateCommandState(commandID);
  360.     }
  361.   }
  362. }
  363.  
  364. //-----------------------------------------------------------------------------------
  365. function goDoCommandParams(command, params)
  366. {
  367.   try
  368.   {
  369.     var controller = top.document.commandDispatcher.getControllerForCommand(command);
  370.     if (controller && controller.isCommandEnabled(command))
  371.     {
  372.       if (controller instanceof Components.interfaces.nsICommandController)
  373.       {
  374.         controller.doCommandWithParams(command, params);
  375.  
  376.         // the following two lines should be removed when we implement observers
  377.         if (params)
  378.           controller.getCommandStateWithParams(command, params);
  379.       }
  380.       else
  381.       {
  382.         controller.doCommand(command);
  383.       }
  384.       ResetStructToolbar();
  385.     }
  386.   }
  387.   catch (e)
  388.   {
  389.     dump("An error occurred executing the "+command+" command\n");
  390.   }
  391. }
  392.  
  393. function pokeStyleUI(uiID, aDesiredState)
  394. {
  395.  try {
  396.   var commandNode = top.document.getElementById(uiID);
  397.   if (!commandNode)
  398.     return;
  399.  
  400.   var uiState = ("true" == commandNode.getAttribute("state"));
  401.   if (aDesiredState != uiState)
  402.   {
  403.     var newState;
  404.     if (aDesiredState)
  405.       newState = "true";
  406.     else
  407.       newState = "false";
  408.     commandNode.setAttribute("state", newState);
  409.   }
  410.  } catch(e) { dump("poking UI for "+uiID+" failed: "+e+"\n"); }
  411. }
  412.  
  413. function doStyleUICommand(cmdStr)
  414. {
  415.   try
  416.   {
  417.     var cmdParams = newCommandParams();
  418.     goDoCommandParams(cmdStr, cmdParams);
  419.     if (cmdParams)
  420.       pokeStyleUI(cmdStr, cmdParams.getBooleanValue("state_all"));
  421.  
  422.     ResetStructToolbar();
  423.   } catch(e) {}
  424. }
  425.  
  426. function pokeMultiStateUI(uiID, cmdParams)
  427. {
  428.   try
  429.   {
  430.     var commandNode = document.getElementById(uiID);
  431.     if (!commandNode)
  432.       return;
  433.  
  434.     var isMixed = cmdParams.getBooleanValue("state_mixed");
  435.     var desiredAttrib;
  436.     if (isMixed)
  437.       desiredAttrib = "mixed";
  438.     else
  439.       desiredAttrib = cmdParams.getCStringValue("state_attribute");
  440.  
  441.     var uiState = commandNode.getAttribute("state");
  442.     if (desiredAttrib != uiState)
  443.     {
  444.       commandNode.setAttribute("state", desiredAttrib);
  445.     }
  446.   } catch(e) {}
  447. }
  448.  
  449. function doStatefulCommand(commandID, newState)
  450. {
  451.   var commandNode = document.getElementById(commandID);
  452.   if (commandNode)
  453.       commandNode.setAttribute("state", newState);
  454.   gContentWindow.focus();   // needed for command dispatch to work
  455.  
  456.   try
  457.   {
  458.     var cmdParams = newCommandParams();
  459.     if (!cmdParams) return;
  460.  
  461.     cmdParams.setCStringValue("state_attribute", newState);
  462.     goDoCommandParams(commandID, cmdParams);
  463.  
  464.     pokeMultiStateUI(commandID, cmdParams);
  465.  
  466.     ResetStructToolbar();
  467.   } catch(e) { dump("error thrown in doStatefulCommand: "+e+"\n"); }
  468. }
  469.  
  470. //-----------------------------------------------------------------------------------
  471. function PrintObject(obj)
  472. {
  473.   dump("-----" + obj + "------\n");
  474.   var names = "";
  475.   for (var i in obj)
  476.   {
  477.     if (i == "value")
  478.       names += i + ": " + obj.value + "\n";
  479.     else if (i == "id")
  480.       names += i + ": " + obj.id + "\n";
  481.     else
  482.       names += i + "\n";
  483.   }
  484.   
  485.   dump(names + "-----------\n");
  486. }
  487.  
  488. //-----------------------------------------------------------------------------------
  489. function PrintNodeID(id)
  490. {
  491.   PrintObject(document.getElementById(id));
  492. }
  493.  
  494. //-----------------------------------------------------------------------------------
  495. var nsDummyHTMLCommand =
  496. {
  497.   isCommandEnabled: function(aCommand, dummy)
  498.   {
  499.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  500.   },
  501.  
  502.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  503.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  504.  
  505.   doCommand: function(aCommand)
  506.   {
  507.     // do nothing
  508.     dump("Hey, who's calling the dummy command?\n");
  509.   }
  510.  
  511. };
  512.  
  513. function openFileInEditor(aCommand, newTab)
  514.   {
  515.     var fp = Components.classes["@mozilla.org/filepicker;1"].createInstance(nsIFilePicker);
  516.     fp.init(window, GetString("OpenHTMLFile"), nsIFilePicker.modeOpen);
  517.  
  518.     SetFilePickerDirectory(fp, "html");
  519.  
  520.     // When loading into Composer, direct user to prefer HTML files and text files,
  521.     //   so we call separately to control the order of the filter list
  522.     fp.appendFilters(nsIFilePicker.filterHTML);
  523.     fp.appendFilters(nsIFilePicker.filterHTMLTemplates);
  524.     fp.appendFilters(nsIFilePicker.filterText);
  525.     fp.appendFilters(nsIFilePicker.filterAll);
  526.  
  527.     /* doesn't handle *.shtml files */
  528.     try {
  529.       fp.show();
  530.       /* need to handle cancel (uncaught exception at present) */
  531.     }
  532.     catch (ex) {
  533.       dump("filePicker.chooseInputFile threw an exception\n");
  534.     }
  535.   
  536.     /* This checks for already open window and activates it... 
  537.      * note that we have to test the native path length
  538.      *  since file.URL will be "file:///" if no filename picked (Cancel button used)
  539.      */
  540.     if (fp.file && fp.file.path.length > 0) {
  541.       SaveFilePickerDirectory(fp, "html");
  542.       editPage(fp.fileURL.spec, window, false, newTab);
  543.     }
  544.   }
  545.  
  546. //-----------------------------------------------------------------------------------
  547. var nsOpenCommand =
  548. {
  549.   isCommandEnabled: function(aCommand, dummy)
  550.   {
  551.     return true;    // we can always do this
  552.   },
  553.  
  554.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  555.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  556.  
  557.   doCommand: function(aCommand)
  558.   {
  559.     openFileInEditor(aCommand, false);
  560.   }
  561. };
  562.  
  563. //-----------------------------------------------------------------------------------
  564. var nsOpenInTabCommand =
  565. {
  566.   isCommandEnabled: function(aCommand, dummy)
  567.   {
  568.     return true;    // we can always do this
  569.   },
  570.  
  571.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  572.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  573.  
  574.   doCommand: function(aCommand)
  575.   {
  576.     openFileInEditor(aCommand, true);
  577.   }
  578. };
  579.  
  580. // STRUCTURE TOOLBAR
  581. //
  582. var nsUpdateStructToolbarCommand =
  583. {
  584.   isCommandEnabled: function(aCommand, dummy)
  585.   {
  586.     UpdateStructToolbar();
  587.     return true;
  588.   },
  589.  
  590.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  591.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  592.   doCommand: function(aCommand)  {}
  593. }
  594.  
  595. // ******* File output commands and utilities ******** //
  596. //-----------------------------------------------------------------------------------
  597. var nsSaveCommand =
  598. {
  599.   isCommandEnabled: function(aCommand, dummy)
  600.   {
  601.     // Always allow saving when editing a remote document,
  602.     //  otherwise the document modified state would prevent that
  603.     //  when you first open a remote file.
  604.     try {
  605.       var docUrl = GetDocumentUrl();
  606.       return IsDocumentEditable() &&
  607.         (IsDocumentModified() || IsHTMLSourceChanged() ||
  608.          IsUrlAboutBlank(docUrl) || GetScheme(docUrl) != "file");
  609.     } catch (e) {return false;}
  610.   },
  611.   
  612.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  613.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  614.  
  615.   doCommand: function(aCommand)
  616.   {
  617.     var result = false;
  618.     var editor = GetCurrentEditor();
  619.     if (editor)
  620.     {
  621.       FinishHTMLSource();
  622.       result = SaveDocument(IsUrlAboutBlank(GetDocumentUrl()), false, editor.contentsMIMEType);
  623.       window.content.focus();
  624.     }
  625.     return result;
  626.   }
  627. }
  628.  
  629. var nsSaveAsCommand =
  630. {
  631.   isCommandEnabled: function(aCommand, dummy)
  632.   {
  633.     return (IsDocumentEditable());
  634.   },
  635.  
  636.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  637.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  638.  
  639.   doCommand: function(aCommand)
  640.   {
  641.     var editor = GetCurrentEditor();
  642.     if (editor)
  643.     {
  644.       FinishHTMLSource();
  645.       var result = SaveDocument(true, false, editor.contentsMIMEType);
  646.       window.content.focus();
  647.       return result;
  648.     }
  649.     return false;
  650.   }
  651. }
  652.  
  653. var nsExportToTextCommand =
  654. {
  655.   isCommandEnabled: function(aCommand, dummy)
  656.   {
  657.     return (IsDocumentEditable());
  658.   },
  659.  
  660.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  661.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  662.  
  663.   doCommand: function(aCommand)
  664.   {
  665.     if (GetCurrentEditor())
  666.     {
  667.       FinishHTMLSource();
  668.       var result = SaveDocument(true, true, "text/plain");
  669.       window.content.focus();
  670.       return result;
  671.     }
  672.     return false;
  673.   }
  674. }
  675.  
  676. var nsSaveAndChangeEncodingCommand =
  677. {
  678.   isCommandEnabled: function(aCommand, dummy)
  679.   {
  680.     return (IsDocumentEditable());
  681.   },
  682.  
  683.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  684.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  685.  
  686.   doCommand: function(aCommand)
  687.   {    
  688.     FinishHTMLSource();
  689.     window.ok = false;
  690.     window.exportToText = false;
  691.     var oldTitle = GetDocumentTitle();
  692.     window.openDialog("chrome://editor/content/EditorSaveAsCharset.xul","_blank", "chrome,close,titlebar,modal,resizable=yes");
  693.  
  694.     if (GetDocumentTitle() != oldTitle)
  695.       UpdateWindowTitle();
  696.  
  697.     if (window.ok)
  698.     {
  699.       if (window.exportToText)
  700.       {
  701.         window.ok = SaveDocument(true, true, "text/plain");
  702.       }
  703.       else
  704.       {
  705.         var editor = GetCurrentEditor();
  706.         window.ok = SaveDocument(true, false, editor ? editor.contentsMIMEType : null);
  707.       }
  708.     }
  709.  
  710.     window.content.focus();
  711.     return window.ok;
  712.   }
  713. };
  714.  
  715. var nsPublishCommand =
  716. {
  717.   isCommandEnabled: function(aCommand, dummy)
  718.   {
  719.     if (IsDocumentEditable())
  720.     {
  721.       // Always allow publishing when editing a local document,
  722.       //  otherwise the document modified state would prevent that
  723.       //  when you first open any local file.
  724.       try {
  725.         var docUrl = GetDocumentUrl();
  726.         return IsDocumentModified() || IsHTMLSourceChanged()
  727.                || IsUrlAboutBlank(docUrl) || GetScheme(docUrl) == "file";
  728.       } catch (e) {return false;}
  729.     }
  730.     return false;
  731.   },
  732.   
  733.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  734.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  735.  
  736.   doCommand: function(aCommand)
  737.   {
  738.     if (GetCurrentEditor())
  739.     {
  740.       var docUrl = GetDocumentUrl();
  741.       var filename = GetFilename(docUrl);
  742.       var publishData;
  743.       var showPublishDialog = false;
  744.  
  745.       // First check pref to always show publish dialog
  746.       try {
  747.         var prefs = GetPrefs();
  748.         if (prefs)
  749.           showPublishDialog = prefs.getBoolPref("editor.always_show_publish_dialog");
  750.       } catch(e) {}
  751.  
  752.       if (!showPublishDialog && filename)
  753.       {
  754.         // Try to get publish data from the document url
  755.         publishData = CreatePublishDataFromUrl(docUrl);
  756.  
  757.         // If none, use default publishing site? Need a pref for this
  758.         //if (!publishData)
  759.         //  publishData = GetPublishDataFromSiteName(GetDefaultPublishSiteName(), filename);
  760.       }
  761.  
  762.       if (showPublishDialog || !publishData)
  763.       {
  764.         // Show the publish dialog
  765.         var publishData = {};
  766.         window.ok = false;
  767.         var oldTitle = GetDocumentTitle();
  768.         window.openDialog("chrome://editor/content/EditorPublish.xul","_blank", 
  769.                           "chrome,close,titlebar,modal", "", "", publishData);
  770.         if (GetDocumentTitle() != oldTitle)
  771.           UpdateWindowTitle();
  772.  
  773.         window.content.focus();
  774.         if (!window.ok)
  775.           return false;
  776.       }
  777.       if (publishData)
  778.       {
  779.         FinishHTMLSource();
  780.         return Publish(publishData);
  781.       }
  782.     }
  783.     return false;
  784.   }
  785. }
  786.  
  787. var nsPublishAsCommand =
  788. {
  789.   isCommandEnabled: function(aCommand, dummy)
  790.   {
  791.     return (IsDocumentEditable());
  792.   },
  793.   
  794.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  795.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  796.  
  797.   doCommand: function(aCommand)
  798.   {
  799.     if (GetCurrentEditor())
  800.     {
  801.       FinishHTMLSource();
  802.  
  803.       window.ok = false;
  804.       publishData = {};
  805.       var oldTitle = GetDocumentTitle();
  806.       window.openDialog("chrome://editor/content/EditorPublish.xul","_blank", 
  807.                         "chrome,close,titlebar,modal", "", "", publishData);
  808.       if (GetDocumentTitle() != oldTitle)
  809.         UpdateWindowTitle();
  810.  
  811.       window.content.focus();
  812.       if (window.ok)
  813.         return Publish(publishData);
  814.     }
  815.     return false;
  816.   }
  817. }
  818.  
  819. // ------- output utilites   ----- //
  820.  
  821. // returns a fileExtension string
  822. function GetExtensionBasedOnMimeType(aMIMEType)
  823. {
  824.   if (CurrentDocumentIsTemplate())
  825.     return "mzt";
  826.  
  827.   try {
  828.     var mimeService = null;
  829.     mimeService = Components.classes["@mozilla.org/mime;1"].getService();
  830.     mimeService = mimeService.QueryInterface(Components.interfaces.nsIMIMEService);
  831.  
  832.     var fileExtension = mimeService.getPrimaryExtension(aMIMEType, null);
  833.  
  834.     // the MIME service likes to give back ".htm" for text/html files,
  835.     // so do a special-case fix here.
  836.     if (fileExtension == "htm")
  837.       fileExtension = "html";
  838.  
  839.     return fileExtension;
  840.   }
  841.   catch (e) {}
  842.   return "";
  843. }
  844.  
  845. function GetSuggestedFileName(aDocumentURLString, aMIMEType)
  846. {
  847.   var extension = GetExtensionBasedOnMimeType(aMIMEType);
  848.   if (extension)
  849.     extension = "." + extension;
  850.  
  851.   // check for existing file name we can use
  852.   if (aDocumentURLString.length >= 0 && !IsUrlAboutBlank(aDocumentURLString))
  853.   {
  854.     var docURI = null;
  855.     try {
  856.  
  857.       var ioService = GetIOService();
  858.       docURI = ioService.newURI(aDocumentURLString, GetCurrentEditor().documentCharacterSet, null);
  859.       docURI = docURI.QueryInterface(Components.interfaces.nsIURL);
  860.  
  861.       // grab the file name
  862.       var url = docURI.fileBaseName;
  863.       if (url)
  864.         return url+extension;
  865.     } catch(e) {}
  866.   } 
  867.  
  868.   // check if there is a title we can use
  869.   var title = GetDocumentTitle();
  870.   // generate a valid filename, if we can't just go with "untitled"
  871.   return GenerateValidFilename(title, extension) || GetString("untitled") + extension;
  872. }
  873.  
  874. // returns file picker result
  875. function PromptForSaveLocation(aDoSaveAsText, aEditorType, aMIMEType, aDocumentURLString)
  876. {
  877.   var dialogResult = {};
  878.   dialogResult.filepickerClick = nsIFilePicker.returnCancel;
  879.   dialogResult.resultingURI = "";
  880.   dialogResult.resultingLocalFile = null;
  881.  
  882.   var fp = null;
  883.   try {
  884.     fp = Components.classes["@mozilla.org/filepicker;1"].createInstance(nsIFilePicker);
  885.   } catch (e) {}
  886.   if (!fp) return dialogResult;
  887.  
  888.   // determine prompt string based on type of saving we'll do
  889.   var promptString;
  890.   if (aDoSaveAsText || aEditorType == "text")
  891.     promptString = GetString("ExportToText");
  892.   else
  893.     promptString = GetString("SaveDocumentAs")
  894.  
  895.   fp.init(window, promptString, nsIFilePicker.modeSave);
  896.  
  897.   // Set filters according to the type of output
  898.   if (aDoSaveAsText)
  899.     fp.appendFilters(nsIFilePicker.filterText);
  900.   else if (CurrentDocumentIsTemplate())
  901.     fp.appendFilters(nsIFilePicker.filterHTMLTemplates);
  902.   else
  903.     fp.appendFilters(nsIFilePicker.filterHTML);
  904.   fp.appendFilters(nsIFilePicker.filterAll);
  905.  
  906.   // now let's actually set the filepicker's suggested filename
  907.   var suggestedFileName = GetSuggestedFileName(aDocumentURLString, aMIMEType);
  908.   if (suggestedFileName)
  909.     fp.defaultString = suggestedFileName;
  910.  
  911.   // set the file picker's current directory
  912.   // assuming we have information needed (like prior saved location)
  913.   try {
  914.     var ioService = GetIOService();
  915.     var fileHandler = GetFileProtocolHandler();
  916.     
  917.     var isLocalFile = true;
  918.     try {
  919.       var docURI = ioService.newURI(aDocumentURLString, GetCurrentEditor().documentCharacterSet, null);
  920.       isLocalFile = docURI.schemeIs("file");
  921.     }
  922.     catch (e) {}
  923.  
  924.     var parentLocation = null;
  925.     if (isLocalFile)
  926.     {
  927.       var fileLocation = fileHandler.getFileFromURLSpec(aDocumentURLString); // this asserts if url is not local
  928.       parentLocation = fileLocation.parent;
  929.     }
  930.     if (parentLocation)
  931.     {
  932.       // Save current filepicker's default location
  933.       if ("gFilePickerDirectory" in window)
  934.         gFilePickerDirectory = fp.displayDirectory;
  935.  
  936.       fp.displayDirectory = parentLocation;
  937.     }
  938.     else
  939.     {
  940.       // Initialize to the last-used directory for the particular type (saved in prefs)
  941.       SetFilePickerDirectory(fp, aEditorType);
  942.     }
  943.   }
  944.   catch(e) {}
  945.  
  946.   dialogResult.filepickerClick = fp.show();
  947.   if (dialogResult.filepickerClick != nsIFilePicker.returnCancel)
  948.   {
  949.     // reset urlstring to new save location
  950.     dialogResult.resultingURIString = fileHandler.getURLSpecFromFile(fp.file);
  951.     dialogResult.resultingLocalFile = fp.file;
  952.     SaveFilePickerDirectory(fp, aEditorType);
  953.   }
  954.   else if ("gFilePickerDirectory" in window && gFilePickerDirectory)
  955.     fp.displayDirectory = gFilePickerDirectory; 
  956.  
  957.   return dialogResult;
  958. }
  959.  
  960. // returns a boolean (whether to continue (true) or not (false) because user canceled)
  961. function PromptAndSetTitleIfNone()
  962. {
  963.   if (GetDocumentTitle()) // we have a title; no need to prompt!
  964.     return true;
  965.  
  966.   var promptService = GetPromptService();
  967.   if (!promptService) return false;
  968.  
  969.   var result = {value:null};
  970.   var captionStr = GetString("DocumentTitle");
  971.   var msgStr = GetString("NeedDocTitle") + '\n' + GetString("DocTitleHelp");
  972.   var confirmed = promptService.prompt(window, captionStr, msgStr, result, null, {value:0});
  973.   if (confirmed)
  974.     SetDocumentTitle(TrimString(result.value));
  975.  
  976.   return confirmed;
  977. }
  978.  
  979. var gPersistObj;
  980.  
  981. // Don't forget to do these things after calling OutputFileWithPersistAPI:
  982. // we need to update the uri before notifying listeners
  983. //    if (doUpdateURI)
  984. //      SetDocumentURI(docURI);
  985. //    UpdateWindowTitle();
  986. //    if (!aSaveCopy)
  987. //      editor.resetModificationCount();
  988.       // this should cause notification to listeners that document has changed
  989.  
  990. const webPersist = Components.interfaces.nsIWebBrowserPersist;
  991. function OutputFileWithPersistAPI(editorDoc, aDestinationLocation, aRelatedFilesParentDir, aMimeType)
  992. {
  993.   gPersistObj = null;
  994.   var editor = GetCurrentEditor();
  995.   try {
  996.     var imeEditor = editor.QueryInterface(Components.interfaces.nsIEditorIMESupport);
  997.     imeEditor.ForceCompositionEnd();
  998.     } catch (e) {}
  999.  
  1000.   var isLocalFile = false;
  1001.   try {
  1002.     var tmp1 = aDestinationLocation.QueryInterface(Components.interfaces.nsIFile);
  1003.     isLocalFile = true;
  1004.   } 
  1005.   catch (e) {
  1006.     try {
  1007.       var tmp = aDestinationLocation.QueryInterface(Components.interfaces.nsIURI);
  1008.       isLocalFile = tmp.schemeIs("file");
  1009.     }
  1010.     catch (e) {}
  1011.   }
  1012.  
  1013.   try {
  1014.     // we should supply a parent directory if/when we turn on functionality to save related documents
  1015.     var persistObj = Components.classes["@mozilla.org/embedding/browser/nsWebBrowserPersist;1"].createInstance(webPersist);
  1016.     persistObj.progressListener = gEditorOutputProgressListener;
  1017.     
  1018.     var wrapColumn = GetWrapColumn();
  1019.     var outputFlags = GetOutputFlags(aMimeType, wrapColumn);
  1020.  
  1021.     // for 4.x parity as well as improving readability of file locally on server
  1022.     // this will always send crlf for upload (http/ftp)
  1023.     if (!isLocalFile) // if we aren't saving locally then send both cr and lf
  1024.     {
  1025.       outputFlags |= webPersist.ENCODE_FLAGS_CR_LINEBREAKS | webPersist.ENCODE_FLAGS_LF_LINEBREAKS;
  1026.  
  1027.       // we want to serialize the output for all remote publishing
  1028.       // some servers can handle only one connection at a time
  1029.       // some day perhaps we can make this user-configurable per site?
  1030.       persistObj.persistFlags = persistObj.persistFlags | webPersist.PERSIST_FLAGS_SERIALIZE_OUTPUT;
  1031.     }
  1032.  
  1033.     // note: we always want to set the replace existing files flag since we have
  1034.     // already given user the chance to not replace an existing file (file picker)
  1035.     // or the user picked an option where the file is implicitly being replaced (save)
  1036.     persistObj.persistFlags = persistObj.persistFlags 
  1037.                             | webPersist.PERSIST_FLAGS_NO_BASE_TAG_MODIFICATIONS
  1038.                             | webPersist.PERSIST_FLAGS_REPLACE_EXISTING_FILES
  1039.                             | webPersist.PERSIST_FLAGS_DONT_FIXUP_LINKS
  1040.                             | webPersist.PERSIST_FLAGS_DONT_CHANGE_FILENAMES
  1041.                             | webPersist.PERSIST_FLAGS_FIXUP_ORIGINAL_DOM;
  1042.     persistObj.saveDocument(editorDoc, aDestinationLocation, aRelatedFilesParentDir, 
  1043.                             aMimeType, outputFlags, wrapColumn);
  1044.     gPersistObj = persistObj;
  1045.   }
  1046.   catch(e) { dump("caught an error, bail\n"); return false; }
  1047.  
  1048.   return true;
  1049. }
  1050.  
  1051. // returns output flags based on mimetype, wrapCol and prefs
  1052. function GetOutputFlags(aMimeType, aWrapColumn)
  1053. {
  1054.   var outputFlags = 0;
  1055.   var editor = GetCurrentEditor();
  1056.   var outputEntity = (editor && editor.documentCharacterSet == "ISO-8859-1")
  1057.     ? webPersist.ENCODE_FLAGS_ENCODE_LATIN1_ENTITIES
  1058.     : webPersist.ENCODE_FLAGS_ENCODE_BASIC_ENTITIES;
  1059.   if (aMimeType == "text/plain")
  1060.   {
  1061.     // When saving in "text/plain" format, always do formatting
  1062.     outputFlags |= webPersist.ENCODE_FLAGS_FORMATTED;
  1063.   }
  1064.   else
  1065.   {
  1066.     try {
  1067.       // Should we prettyprint? Check the pref
  1068.       var prefs = GetPrefs();
  1069.       if (prefs.getBoolPref("editor.prettyprint"))
  1070.         outputFlags |= webPersist.ENCODE_FLAGS_FORMATTED;
  1071.  
  1072.       // How much entity names should we output? Check the pref
  1073.       var encodeEntity = prefs.getCharPref("editor.encode_entity");
  1074.       switch (encodeEntity) {
  1075.         case "basic"  : outputEntity = webPersist.ENCODE_FLAGS_ENCODE_BASIC_ENTITIES; break;
  1076.         case "latin1" : outputEntity = webPersist.ENCODE_FLAGS_ENCODE_LATIN1_ENTITIES; break;
  1077.         case "html"   : outputEntity = webPersist.ENCODE_FLAGS_ENCODE_HTML_ENTITIES; break;
  1078.         case "none"   : outputEntity = 0; break;
  1079.       }
  1080.     }
  1081.     catch (e) {}
  1082.   }
  1083.   outputFlags |= outputEntity;
  1084.  
  1085.   if (aWrapColumn > 0)
  1086.     outputFlags |= webPersist.ENCODE_FLAGS_WRAP;
  1087.  
  1088.   return outputFlags;
  1089. }
  1090.  
  1091. // returns number of column where to wrap
  1092. const nsIWebBrowserPersist = Components.interfaces.nsIWebBrowserPersist;
  1093. function GetWrapColumn()
  1094. {
  1095.   try {
  1096.     return GetCurrentEditor().wrapWidth;
  1097.   } catch (e) {}
  1098.   return 0;
  1099. }
  1100.  
  1101. function GetPromptService()
  1102. {
  1103.   var promptService;
  1104.   try {
  1105.     promptService = Components.classes["@mozilla.org/embedcomp/prompt-service;1"].getService();
  1106.     promptService = promptService.QueryInterface(Components.interfaces.nsIPromptService);
  1107.   }
  1108.   catch (e) {}
  1109.   return promptService;
  1110. }
  1111.  
  1112. const gShowDebugOutputStateChange = false;
  1113. const gShowDebugOutputProgress = false;
  1114. const gShowDebugOutputStatusChange = false;
  1115.  
  1116. const gShowDebugOutputLocationChange = false;
  1117. const gShowDebugOutputSecurityChange = false;
  1118.  
  1119. const nsIWebProgressListener = Components.interfaces.nsIWebProgressListener;
  1120. const nsIChannel = Components.interfaces.nsIChannel;
  1121.  
  1122. const kErrorBindingAborted = 2152398850;
  1123. const kErrorBindingRedirected = 2152398851;
  1124. const kFileNotFound = 2152857618;
  1125.  
  1126. var gEditorOutputProgressListener =
  1127. {
  1128.   onStateChange : function(aWebProgress, aRequest, aStateFlags, aStatus)
  1129.   {
  1130.     var editor = GetCurrentEditor();
  1131.  
  1132.     // Use this to access onStateChange flags
  1133.     var requestSpec;
  1134.     try {
  1135.       var channel = aRequest.QueryInterface(nsIChannel);
  1136.       requestSpec = StripUsernamePasswordFromURI(channel.URI);
  1137.     } catch (e) {
  1138.       if ( gShowDebugOutputStateChange)
  1139.         dump("***** onStateChange; NO REQUEST CHANNEL\n");
  1140.     }
  1141.  
  1142.     var pubSpec;
  1143.     if (gPublishData)
  1144.       pubSpec = gPublishData.publishUrl + gPublishData.docDir + gPublishData.filename;
  1145.  
  1146.     if (gShowDebugOutputStateChange)
  1147.     {
  1148.       dump("\n***** onStateChange request: " + requestSpec + "\n");
  1149.       dump("      state flags: ");
  1150.  
  1151.       if (aStateFlags & nsIWebProgressListener.STATE_START)
  1152.         dump(" STATE_START, ");
  1153.       if (aStateFlags & nsIWebProgressListener.STATE_STOP)
  1154.         dump(" STATE_STOP, ");
  1155.       if (aStateFlags & nsIWebProgressListener.STATE_IS_NETWORK)
  1156.         dump(" STATE_IS_NETWORK ");
  1157.  
  1158.       dump("\n * requestSpec="+requestSpec+", pubSpec="+pubSpec+", aStatus="+aStatus+"\n");
  1159.  
  1160.       DumpDebugStatus(aStatus);
  1161.     }
  1162.     // The rest only concerns publishing, so bail out if no dialog
  1163.     if (!gProgressDialog)
  1164.       return;
  1165.  
  1166.     // Detect start of file upload of any file:
  1167.     // (We ignore any START messages after gPersistObj says publishing is finished
  1168.     if ((aStateFlags & nsIWebProgressListener.STATE_START)
  1169.          && gPersistObj && requestSpec
  1170.          && (gPersistObj.currentState != gPersistObj.PERSIST_STATE_FINISHED))
  1171.     {
  1172.       try {
  1173.         // Add url to progress dialog's list showing each file uploading
  1174.         gProgressDialog.SetProgressStatus(GetFilename(requestSpec), "busy");
  1175.       } catch(e) {}
  1176.     }
  1177.  
  1178.     // Detect end of file upload of any file:
  1179.     if (aStateFlags & nsIWebProgressListener.STATE_STOP)
  1180.     {
  1181.       // ignore aStatus == kErrorBindingAborted; check http response for possible errors
  1182.       try {
  1183.         // check http channel for response: 200 range is ok; other ranges are not
  1184.         var httpChannel = aRequest.QueryInterface(Components.interfaces.nsIHttpChannel);
  1185.         var httpResponse = httpChannel.responseStatus;
  1186.         if (httpResponse < 200 || httpResponse >= 300)
  1187.           aStatus = httpResponse;   // not a real error but enough to pass check below
  1188.         else if (aStatus == kErrorBindingAborted)
  1189.           aStatus = 0;
  1190.  
  1191.         if (gShowDebugOutputStateChange)
  1192.           dump("http response is: "+httpResponse+"\n");
  1193.       } 
  1194.       catch(e) 
  1195.       {
  1196.         if (aStatus == kErrorBindingAborted)
  1197.           aStatus = 0;
  1198.       }
  1199.  
  1200.       // We abort publishing for all errors except if image src file is not found
  1201.       var abortPublishing = (aStatus != 0 && aStatus != kFileNotFound);
  1202.  
  1203.       // Notify progress dialog when we receive the STOP
  1204.       //  notification for a file if there was an error 
  1205.       //  or a successful finish
  1206.       //  (Check requestSpec to be sure message is for destination url)
  1207.       if (aStatus != 0 
  1208.            || (requestSpec && requestSpec.indexOf(GetScheme(gPublishData.publishUrl)) == 0))
  1209.       {
  1210.         try {
  1211.           gProgressDialog.SetProgressFinished(GetFilename(requestSpec), aStatus);
  1212.         } catch(e) {}
  1213.       }
  1214.  
  1215.  
  1216.       if (abortPublishing)
  1217.       {
  1218.         // Cancel publishing
  1219.         gPersistObj.cancelSave();
  1220.  
  1221.         // Don't do any commands after failure
  1222.         gCommandAfterPublishing = null;
  1223.  
  1224.         // Restore original document to undo image src url adjustments
  1225.         if (gRestoreDocumentSource)
  1226.         {
  1227.           try {
  1228.             editor.rebuildDocumentFromSource(gRestoreDocumentSource);
  1229.  
  1230.             // Clear transaction cache since we just did a potentially 
  1231.             //  very large insert and this will eat up memory
  1232.             editor.transactionManager.clear();
  1233.           }
  1234.           catch (e) {}
  1235.         }
  1236.  
  1237.         // Notify progress dialog that we're finished
  1238.         //  and keep open to show error
  1239.         gProgressDialog.SetProgressFinished(null, 0);
  1240.  
  1241.         // We don't want to change location or reset mod count, etc.
  1242.         return;
  1243.       }
  1244.  
  1245.       //XXX HACK: "file://" protocol is not supported in network code
  1246.       //    (bug 151867 filed to add this support, bug 151869 filed
  1247.       //     to remove this and other code in nsIWebBrowserPersist)
  1248.       //    nsIWebBrowserPersist *does* copy the file(s), but we don't 
  1249.       //    get normal onStateChange messages.
  1250.  
  1251.       // Case 1: If images are included, we get fairly normal
  1252.       //    STATE_START/STATE_STOP & STATE_IS_NETWORK messages associated with the image files,
  1253.       //    thus we must finish HTML file progress below
  1254.  
  1255.       // Case 2: If just HTML file is uploaded, we get STATE_START and STATE_STOP
  1256.       //    notification with a null "requestSpec", and 
  1257.       //    the gPersistObj is destroyed before we get here!
  1258.       //    So create an new object so we can flow through normal processing below
  1259.       if (!requestSpec && GetScheme(gPublishData.publishUrl) == "file"
  1260.           && (!gPersistObj || gPersistObj.currentState == nsIWebBrowserPersist.PERSIST_STATE_FINISHED))
  1261.       {
  1262.         aStateFlags |= nsIWebProgressListener.STATE_IS_NETWORK;
  1263.         if (!gPersistObj)
  1264.         {          
  1265.           gPersistObj =
  1266.           {
  1267.             result : aStatus,
  1268.             currentState : nsIWebBrowserPersist.PERSIST_STATE_FINISHED
  1269.           }
  1270.         }
  1271.       }
  1272.  
  1273.       // STATE_IS_NETWORK signals end of publishing, as does the gPersistObj.currentState
  1274.       if (aStateFlags & nsIWebProgressListener.STATE_IS_NETWORK
  1275.           && gPersistObj.currentState == nsIWebBrowserPersist.PERSIST_STATE_FINISHED)
  1276.       {
  1277.         if (GetScheme(gPublishData.publishUrl) == "file")
  1278.         {
  1279.           //XXX "file://" hack: We don't get notified about the HTML file, so end progress for it
  1280.           // (This covers both "Case 1 and 2" described above)
  1281.           gProgressDialog.SetProgressFinished(gPublishData.filename, gPersistObj.result);
  1282.         }
  1283.  
  1284.         if (gPersistObj.result == 0)
  1285.         {
  1286.           // All files are finished and publishing succeeded (some images may have failed)
  1287.           try {
  1288.             // Make a new docURI from the "browse location" in case "publish location" was FTP
  1289.             // We need to set document uri before notifying listeners
  1290.             var docUrl = GetDocUrlFromPublishData(gPublishData);
  1291.             SetDocumentURI(GetIOService().newURI(docUrl, editor.documentCharacterSet, null));
  1292.  
  1293.             UpdateWindowTitle();
  1294.  
  1295.             // this should cause notification to listeners that doc has changed
  1296.             editor.resetModificationCount();
  1297.  
  1298.             // Set UI based on whether we're editing a remote or local url
  1299.             SetSaveAndPublishUI(urlstring);
  1300.  
  1301.           } catch (e) {}
  1302.  
  1303.           // Save publishData to prefs
  1304.           if (gPublishData)
  1305.           {
  1306.             if (gPublishData.savePublishData)
  1307.             {
  1308.               // We published successfully, so we can safely
  1309.               //  save docDir and otherDir to prefs
  1310.               gPublishData.saveDirs = true;
  1311.               SavePublishDataToPrefs(gPublishData);
  1312.             }
  1313.             else
  1314.               SavePassword(gPublishData);
  1315.           }
  1316.  
  1317.           // Ask progress dialog to close, but it may not
  1318.           // if user checked checkbox to keep it open
  1319.           gProgressDialog.RequestCloseDialog();
  1320.         }
  1321.         else
  1322.         {
  1323.           // We previously aborted publishing because of error:
  1324.           //   Calling gPersistObj.cancelSave() resulted in a non-zero gPersistObj.result,
  1325.           //   so notify progress dialog we're finished
  1326.           gProgressDialog.SetProgressFinished(null, 0);
  1327.         }
  1328.       }
  1329.     }
  1330.   },
  1331.  
  1332.   onProgressChange : function(aWebProgress, aRequest, aCurSelfProgress,
  1333.                               aMaxSelfProgress, aCurTotalProgress, aMaxTotalProgress)
  1334.   {
  1335.     if (!gPersistObj)
  1336.       return;
  1337.  
  1338.     if (gShowDebugOutputProgress)
  1339.     {
  1340.       dump("\n onProgressChange: gPersistObj.result="+gPersistObj.result+"\n");
  1341.       try {
  1342.       var channel = aRequest.QueryInterface(nsIChannel);
  1343.       dump("***** onProgressChange request: " + channel.URI.spec + "\n");
  1344.       }
  1345.       catch (e) {}
  1346.       dump("*****       self:  "+aCurSelfProgress+" / "+aMaxSelfProgress+"\n");
  1347.       dump("*****       total: "+aCurTotalProgress+" / "+aMaxTotalProgress+"\n\n");
  1348.  
  1349.       if (gPersistObj.currentState == gPersistObj.PERSIST_STATE_READY)
  1350.         dump(" Persister is ready to save data\n\n");
  1351.       else if (gPersistObj.currentState == gPersistObj.PERSIST_STATE_SAVING)
  1352.         dump(" Persister is saving data.\n\n");
  1353.       else if (gPersistObj.currentState == gPersistObj.PERSIST_STATE_FINISHED)
  1354.         dump(" PERSISTER HAS FINISHED SAVING DATA\n\n\n");
  1355.     }
  1356.   },
  1357.  
  1358.   onLocationChange : function(aWebProgress, aRequest, aLocation)
  1359.   {
  1360.     if (gShowDebugOutputLocationChange)
  1361.     {
  1362.       dump("***** onLocationChange: "+aLocation.spec+"\n");
  1363.       try {
  1364.         var channel = aRequest.QueryInterface(nsIChannel);
  1365.         dump("*****          request: " + channel.URI.spec + "\n");
  1366.       }
  1367.       catch(e) {}
  1368.     }
  1369.   },
  1370.  
  1371.   onStatusChange : function(aWebProgress, aRequest, aStatus, aMessage)
  1372.   {
  1373.     if (gShowDebugOutputStatusChange)
  1374.     {
  1375.       dump("***** onStatusChange: "+aMessage+"\n");
  1376.       try {
  1377.         var channel = aRequest.QueryInterface(nsIChannel);
  1378.         dump("*****        request: " + channel.URI.spec + "\n");
  1379.       }
  1380.       catch (e) { dump("          couldn't get request\n"); }
  1381.       
  1382.       DumpDebugStatus(aStatus);
  1383.  
  1384.       if (gPersistObj)
  1385.       {
  1386.         if(gPersistObj.currentState == gPersistObj.PERSIST_STATE_READY)
  1387.           dump(" Persister is ready to save data\n\n");
  1388.         else if(gPersistObj.currentState == gPersistObj.PERSIST_STATE_SAVING)
  1389.           dump(" Persister is saving data.\n\n");
  1390.         else if(gPersistObj.currentState == gPersistObj.PERSIST_STATE_FINISHED)
  1391.           dump(" PERSISTER HAS FINISHED SAVING DATA\n\n\n");
  1392.       }
  1393.     }
  1394.   },
  1395.  
  1396.   onSecurityChange : function(aWebProgress, aRequest, state)
  1397.   {
  1398.     if (gShowDebugOutputSecurityChange)
  1399.     {
  1400.       try {
  1401.         var channel = aRequest.QueryInterface(nsIChannel);
  1402.         dump("***** onSecurityChange request: " + channel.URI.spec + "\n");
  1403.       } catch (e) {}
  1404.     }
  1405.   },
  1406.  
  1407.   QueryInterface : function(aIID)
  1408.   {
  1409.     if (aIID.equals(Components.interfaces.nsIWebProgressListener)
  1410.     || aIID.equals(Components.interfaces.nsISupports)
  1411.     || aIID.equals(Components.interfaces.nsISupportsWeakReference)
  1412.     || aIID.equals(Components.interfaces.nsIPrompt)
  1413.     || aIID.equals(Components.interfaces.nsIAuthPrompt))
  1414.       return this;
  1415.     throw Components.results.NS_NOINTERFACE;
  1416.   },
  1417.  
  1418. // nsIPrompt
  1419.   alert : function(dlgTitle, text)
  1420.   {
  1421.     AlertWithTitle(dlgTitle, text, gProgressDialog ? gProgressDialog : window);
  1422.   },
  1423.   alertCheck : function(dialogTitle, text, checkBoxLabel, checkObj)
  1424.   {
  1425.     AlertWithTitle(dialogTitle, text);
  1426.   },
  1427.   confirm : function(dlgTitle, text)
  1428.   {
  1429.     return ConfirmWithTitle(dlgTitle, text, null, null);
  1430.   },
  1431.   confirmCheck : function(dlgTitle, text, checkBoxLabel, checkObj)
  1432.   {
  1433.     var promptServ = GetPromptService();
  1434.     if (!promptServ)
  1435.       return;
  1436.  
  1437.     promptServ.confirmEx(window, dlgTitle, text, nsIPromptService.STD_OK_CANCEL_BUTTONS,
  1438.                          "", "", "", checkBoxLabel, checkObj);
  1439.   },
  1440.   confirmEx : function(dlgTitle, text, btnFlags, btn0Title, btn1Title, btn2Title, checkBoxLabel, checkVal)
  1441.   {
  1442.     var promptServ = GetPromptService();
  1443.     if (!promptServ)
  1444.      return 0;
  1445.  
  1446.     return promptServ.confirmEx(window, dlgTitle, text, btnFlags,
  1447.                         btn0Title, btn1Title, btn2Title,
  1448.                         checkBoxLabel, checkVal);
  1449.   },
  1450.   prompt : function(dlgTitle, text, inoutText, checkBoxLabel, checkObj)
  1451.   {
  1452.     var promptServ = GetPromptService();
  1453.     if (!promptServ)
  1454.      return false;
  1455.  
  1456.     return promptServ.prompt(window, dlgTitle, text, inoutText, checkBoxLabel, checkObj);
  1457.   },
  1458.   promptPassword : function(dlgTitle, text, pwObj, checkBoxLabel, savePWObj)
  1459.   {
  1460.  
  1461.     var promptServ = GetPromptService();
  1462.     if (!promptServ)
  1463.      return false;
  1464.  
  1465.     var ret = false;
  1466.     try {
  1467.       // Note difference with nsIAuthPrompt::promptPassword, which has 
  1468.       // just "in" savePassword param, while nsIPrompt is "inout"
  1469.       // Initialize with user's previous preference for this site
  1470.       if (gPublishData)
  1471.         savePWObj.value = gPublishData.savePassword;
  1472.  
  1473.       ret = promptServ.promptPassword(gProgressDialog ? gProgressDialog : window,
  1474.                                       dlgTitle, text, pwObj, checkBoxLabel, savePWObj);
  1475.  
  1476.       if (!ret)
  1477.         setTimeout(CancelPublishing, 0);
  1478.  
  1479.       if (ret && gPublishData)
  1480.         UpdateUsernamePasswordFromPrompt(gPublishData, gPublishData.username, pwObj.value, savePWObj.value);
  1481.     } catch(e) {}
  1482.  
  1483.     return ret;
  1484.   },
  1485.   promptUsernameAndPassword : function(dlgTitle, text, userObj, pwObj, checkBoxLabel, savePWObj)
  1486.   {
  1487.     var ret = PromptUsernameAndPassword(dlgTitle, text, savePWObj.value, userObj, pwObj);
  1488.     if (!ret)
  1489.       setTimeout(CancelPublishing, 0);
  1490.  
  1491.     return ret;
  1492.   },
  1493.   select : function(dlgTitle, text, count, selectList, outSelection)
  1494.   {
  1495.     var promptServ = GetPromptService();
  1496.     if (!promptServ)
  1497.       return false;
  1498.  
  1499.     return promptServ.select(window, dlgTitle, text, count, selectList, outSelection);
  1500.   },
  1501.  
  1502. // nsIAuthPrompt
  1503.   prompt : function(dlgTitle, text, pwrealm, savePW, defaultText, result)
  1504.   {
  1505.     var promptServ = GetPromptService();
  1506.     if (!promptServ)
  1507.       return false;
  1508.  
  1509.     var savePWObj = {value:savePW};
  1510.     var ret = promptServ.prompt(gProgressDialog ? gProgressDialog : window,
  1511.                                 dlgTitle, text, defaultText, pwrealm, savePWObj);
  1512.     if (!ret)
  1513.       setTimeout(CancelPublishing, 0);
  1514.     return ret;
  1515.   },
  1516.  
  1517.   promptUsernameAndPassword : function(dlgTitle, text, pwrealm, savePW, userObj, pwObj)
  1518.   {
  1519.     var ret = PromptUsernameAndPassword(dlgTitle, text, savePW, userObj, pwObj);
  1520.     if (!ret)
  1521.       setTimeout(CancelPublishing, 0);
  1522.     return ret;
  1523.   },
  1524.  
  1525.   promptPassword : function(dlgTitle, text, pwrealm, savePW, pwObj)
  1526.   {
  1527.     var ret = false;
  1528.     try {
  1529.       var promptServ = GetPromptService();
  1530.       if (!promptServ)
  1531.         return false;
  1532.  
  1533.       // Note difference with nsIPrompt::promptPassword, which has 
  1534.       // "inout" savePassword param, while nsIAuthPrompt is just "in"
  1535.       // Also nsIAuth doesn't supply "checkBoxLabel"
  1536.       // Initialize with user's previous preference for this site
  1537.       var savePWObj = {value:savePW};
  1538.       // Initialize with user's previous preference for this site
  1539.       if (gPublishData)
  1540.         savePWObj.value = gPublishData.savePassword;
  1541.  
  1542.       ret = promptServ.promptPassword(gProgressDialog ? gProgressDialog : window,
  1543.                                       dlgTitle, text, pwObj, GetString("SavePassword"), savePWObj);
  1544.  
  1545.       if (!ret)
  1546.         setTimeout(CancelPublishing, 0);
  1547.  
  1548.       if (ret && gPublishData)
  1549.         UpdateUsernamePasswordFromPrompt(gPublishData, gPublishData.username, pwObj.value, savePWObj.value);
  1550.     } catch(e) {}
  1551.  
  1552.     return ret;
  1553.   }
  1554. }
  1555.  
  1556. function PromptUsernameAndPassword(dlgTitle, text, savePW, userObj, pwObj)
  1557. {
  1558.   // HTTP prompts us twice even if user Cancels from 1st attempt!
  1559.   // So never put up dialog if there's no publish data
  1560.   if (!gPublishData)
  1561.     return false
  1562.  
  1563.   var ret = false;
  1564.   try {
  1565.     var promptServ = GetPromptService();
  1566.     if (!promptServ)
  1567.       return false;
  1568.  
  1569.     var savePWObj = {value:savePW};
  1570.  
  1571.     // Initialize with user's previous preference for this site
  1572.     if (gPublishData)
  1573.     {
  1574.       // HTTP put uses this dialog if either username or password is bad,
  1575.       //   so prefill username input field with the previous value for modification
  1576.       savePWObj.value = gPublishData.savePassword;
  1577.       if (!userObj.value)
  1578.         userObj.value = gPublishData.username;
  1579.     }
  1580.  
  1581.     ret = promptServ.promptUsernameAndPassword(gProgressDialog ? gProgressDialog : window, 
  1582.                                                dlgTitle, text, userObj, pwObj, 
  1583.                                                GetString("SavePassword"), savePWObj);
  1584.     if (ret && gPublishData)
  1585.       UpdateUsernamePasswordFromPrompt(gPublishData, userObj.value, pwObj.value, savePWObj.value);
  1586.  
  1587.   } catch (e) {}
  1588.  
  1589.   return ret;
  1590. }
  1591.  
  1592. function DumpDebugStatus(aStatus)
  1593. {
  1594.   // see nsError.h and netCore.h and ftpCore.h
  1595.  
  1596.   if (aStatus == kErrorBindingAborted)
  1597.     dump("***** status is NS_BINDING_ABORTED\n");
  1598.   else if (aStatus == kErrorBindingRedirected)
  1599.     dump("***** status is NS_BINDING_REDIRECTED\n");
  1600.   else if (aStatus == 2152398859) // in netCore.h 11
  1601.     dump("***** status is ALREADY_CONNECTED\n");
  1602.   else if (aStatus == 2152398860) // in netCore.h 12
  1603.     dump("***** status is NOT_CONNECTED\n");
  1604.   else if (aStatus == 2152398861) //  in nsISocketTransportService.idl 13
  1605.     dump("***** status is CONNECTION_REFUSED\n");
  1606.   else if (aStatus == 2152398862) // in nsISocketTransportService.idl 14
  1607.     dump("***** status is NET_TIMEOUT\n");
  1608.   else if (aStatus == 2152398863) // in netCore.h 15
  1609.     dump("***** status is IN_PROGRESS\n");
  1610.   else if (aStatus == 2152398864) // 0x804b0010 in netCore.h 16
  1611.     dump("***** status is OFFLINE\n");
  1612.   else if (aStatus == 2152398865) // in netCore.h 17
  1613.     dump("***** status is NO_CONTENT\n");
  1614.   else if (aStatus == 2152398866) // in netCore.h 18
  1615.     dump("***** status is UNKNOWN_PROTOCOL\n");
  1616.   else if (aStatus == 2152398867) // in netCore.h 19
  1617.     dump("***** status is PORT_ACCESS_NOT_ALLOWED\n");
  1618.   else if (aStatus == 2152398868) // in nsISocketTransportService.idl 20
  1619.     dump("***** status is NET_RESET\n");
  1620.   else if (aStatus == 2152398869) // in ftpCore.h 21
  1621.     dump("***** status is FTP_LOGIN\n");
  1622.   else if (aStatus == 2152398870) // in ftpCore.h 22
  1623.     dump("***** status is FTP_CWD\n");
  1624.   else if (aStatus == 2152398871) // in ftpCore.h 23
  1625.     dump("***** status is FTP_PASV\n");
  1626.   else if (aStatus == 2152398872) // in ftpCore.h 24
  1627.     dump("***** status is FTP_PWD\n");
  1628.   else if (aStatus == 2152857601)
  1629.     dump("***** status is UNRECOGNIZED_PATH\n");
  1630.   else if (aStatus == 2152857602)
  1631.     dump("***** status is UNRESOLABLE SYMLINK\n");
  1632.   else if (aStatus == 2152857604)
  1633.     dump("***** status is UNKNOWN_TYPE\n");
  1634.   else if (aStatus == 2152857605)
  1635.     dump("***** status is DESTINATION_NOT_DIR\n");
  1636.   else if (aStatus == 2152857606)
  1637.     dump("***** status is TARGET_DOES_NOT_EXIST\n");
  1638.   else if (aStatus == 2152857608)
  1639.     dump("***** status is ALREADY_EXISTS\n");
  1640.   else if (aStatus == 2152857609)
  1641.     dump("***** status is INVALID_PATH\n");
  1642.   else if (aStatus == 2152857610)
  1643.     dump("***** status is DISK_FULL\n");
  1644.   else if (aStatus == 2152857612)
  1645.     dump("***** status is NOT_DIRECTORY\n");
  1646.   else if (aStatus == 2152857613)
  1647.     dump("***** status is IS_DIRECTORY\n");
  1648.   else if (aStatus == 2152857614)
  1649.     dump("***** status is IS_LOCKED\n");
  1650.   else if (aStatus == 2152857615)
  1651.     dump("***** status is TOO_BIG\n");
  1652.   else if (aStatus == 2152857616)
  1653.     dump("***** status is NO_DEVICE_SPACE\n");
  1654.   else if (aStatus == 2152857617)
  1655.     dump("***** status is NAME_TOO_LONG\n");
  1656.   else if (aStatus == 2152857618) // 80520012
  1657.     dump("***** status is FILE_NOT_FOUND\n");
  1658.   else if (aStatus == 2152857619)
  1659.     dump("***** status is READ_ONLY\n");
  1660.   else if (aStatus == 2152857620)
  1661.     dump("***** status is DIR_NOT_EMPTY\n");
  1662.   else if (aStatus == 2152857621)
  1663.     dump("***** status is ACCESS_DENIED\n");
  1664.   else if (aStatus == 2152398878)
  1665.     dump("***** status is ? (No connection or time out?)\n");
  1666.   else
  1667.     dump("***** status is " + aStatus + "\n");
  1668. }
  1669.  
  1670. // Update any data that the user supplied in a prompt dialog
  1671. function UpdateUsernamePasswordFromPrompt(publishData, username, password, savePassword)
  1672. {
  1673.   if (!publishData)
  1674.     return;
  1675.   
  1676.   // Set flag to save publish data after publishing if it changed in dialog 
  1677.   //  and the "SavePassword" checkbox was checked
  1678.   //  or we already had site data for this site
  1679.   // (Thus we don't automatically create a site until user brings up Publish As dialog)
  1680.   publishData.savePublishData = (gPublishData.username != username || gPublishData.password != password)
  1681.                                 && (savePassword || !publishData.notInSiteData);
  1682.  
  1683.   publishData.username = username;
  1684.   publishData.password = password;
  1685.   publishData.savePassword = savePassword;
  1686. }
  1687.  
  1688. const kSupportedTextMimeTypes =
  1689. [
  1690.   "text/plain",
  1691.   "text/css",
  1692.   "text/rdf",
  1693.   "text/xsl",
  1694.   "text/javascript",
  1695.   "application/x-javascript",
  1696.   "text/xul",
  1697.   "application/vnd.mozilla.xul+xml"
  1698. ];
  1699.  
  1700. function IsSupportedTextMimeType(aMimeType)
  1701. {
  1702.   for (var i = 0; i < kSupportedTextMimeTypes.length; i++)
  1703.   {
  1704.     if (kSupportedTextMimeTypes[i] == aMimeType)
  1705.       return true;
  1706.   }
  1707.   return false;
  1708. }
  1709.  
  1710. // throws an error or returns true if user attempted save; false if user canceled save
  1711. function SaveDocument(aSaveAs, aSaveCopy, aMimeType)
  1712. {
  1713.   var editor = GetCurrentEditor();
  1714.   if (!aMimeType || aMimeType == "" || !editor)
  1715.     throw NS_ERROR_NOT_INITIALIZED;
  1716.  
  1717.   var editorDoc = editor.document;
  1718.   if (!editorDoc)
  1719.     throw NS_ERROR_NOT_INITIALIZED;
  1720.  
  1721.   // if we don't have the right editor type bail (we handle text and html)
  1722.   var editorType = GetCurrentEditorType();
  1723.   if (editorType != "text" && editorType != "html" 
  1724.       && editorType != "htmlmail" && editorType != "textmail")
  1725.     throw NS_ERROR_NOT_IMPLEMENTED;
  1726.  
  1727.   var saveAsTextFile = IsSupportedTextMimeType(aMimeType);
  1728.  
  1729.   // check if the file is to be saved is a format we don't understand; if so, bail
  1730.   if (aMimeType != "text/html" && !saveAsTextFile)
  1731.     throw NS_ERROR_NOT_IMPLEMENTED;
  1732.  
  1733.   if (saveAsTextFile)
  1734.     aMimeType = "text/plain";
  1735.  
  1736.   var urlstring = GetDocumentUrl();
  1737.   var mustShowFileDialog = (aSaveAs || IsUrlAboutBlank(urlstring) || (urlstring == ""));
  1738.  
  1739.   // If editing a remote URL, force SaveAs dialog
  1740.   if (!mustShowFileDialog && GetScheme(urlstring) != "file")
  1741.     mustShowFileDialog = true;
  1742.  
  1743.   var replacing = !aSaveAs;
  1744.   var titleChanged = false;
  1745.   var doUpdateURI = false;
  1746.   var tempLocalFile = null;
  1747.  
  1748.   if (mustShowFileDialog)
  1749.   {
  1750.       try {
  1751.         // Prompt for title if we are saving to HTML
  1752.         if (!saveAsTextFile && (editorType == "html"))
  1753.         {
  1754.           var userContinuing = PromptAndSetTitleIfNone(); // not cancel
  1755.           if (!userContinuing)
  1756.             return false;
  1757.         }
  1758.  
  1759.         var dialogResult = PromptForSaveLocation(saveAsTextFile, editorType, aMimeType, urlstring);
  1760.         if (dialogResult.filepickerClick == nsIFilePicker.returnCancel)
  1761.           return false;
  1762.  
  1763.         replacing = (dialogResult.filepickerClick == nsIFilePicker.returnReplace);
  1764.         urlstring = dialogResult.resultingURIString;
  1765.         tempLocalFile = dialogResult.resultingLocalFile;
  1766.  
  1767.       // update the new URL for the webshell unless we are saving a copy
  1768.       if (!aSaveCopy)
  1769.         doUpdateURI = true;
  1770.    } catch (e) {  return false; }
  1771.   } // mustShowFileDialog
  1772.  
  1773.   var success = true;
  1774.   var ioService;
  1775.   try {
  1776.     // if somehow we didn't get a local file but we did get a uri, 
  1777.     // attempt to create the localfile if it's a "file" url
  1778.     var docURI;
  1779.     if (!tempLocalFile)
  1780.     {
  1781.       ioService = GetIOService();
  1782.       docURI = ioService.newURI(urlstring, editor.documentCharacterSet, null);
  1783.       
  1784.       if (docURI.schemeIs("file"))
  1785.       {
  1786.         var fileHandler = GetFileProtocolHandler();
  1787.         tempLocalFile = fileHandler.getFileFromURLSpec(urlstring).QueryInterface(Components.interfaces.nsILocalFile);
  1788.       }
  1789.     }
  1790.  
  1791.     // this is the location where the related files will go
  1792.     var relatedFilesDir = null;
  1793.     
  1794.     // First check pref for saving associated files
  1795.     var saveAssociatedFiles = false;
  1796.     try {
  1797.       var prefs = GetPrefs();
  1798.       saveAssociatedFiles = prefs.getBoolPref("editor.save_associated_files");
  1799.     } catch (e) {}
  1800.  
  1801.     // Only change links or move files if pref is set 
  1802.     //  and we are saving to a new location
  1803.     if (saveAssociatedFiles && aSaveAs)
  1804.     {
  1805.       try {
  1806.         if (tempLocalFile)
  1807.         {
  1808.           // if we are saving to the same parent directory, don't set relatedFilesDir
  1809.           // grab old location, chop off file
  1810.           // grab new location, chop off file, compare
  1811.           var oldLocation = GetDocumentUrl();
  1812.           var oldLocationLastSlash = oldLocation.lastIndexOf("\/");
  1813.           if (oldLocationLastSlash != -1)
  1814.             oldLocation = oldLocation.slice(0, oldLocationLastSlash);
  1815.  
  1816.           var relatedFilesDirStr = urlstring;
  1817.           var newLocationLastSlash = relatedFilesDirStr.lastIndexOf("\/");
  1818.           if (newLocationLastSlash != -1)
  1819.             relatedFilesDirStr = relatedFilesDirStr.slice(0, newLocationLastSlash);
  1820.           if (oldLocation == relatedFilesDirStr || IsUrlAboutBlank(oldLocation))
  1821.             relatedFilesDir = null;
  1822.           else
  1823.             relatedFilesDir = tempLocalFile.parent;
  1824.         }
  1825.         else
  1826.         {
  1827.           var lastSlash = urlstring.lastIndexOf("\/");
  1828.           if (lastSlash != -1)
  1829.           {
  1830.             var relatedFilesDirString = urlstring.slice(0, lastSlash + 1);  // include last slash
  1831.             ioService = GetIOService();
  1832.             relatedFilesDir = ioService.newURI(relatedFilesDirString, editor.documentCharacterSet, null);
  1833.           }
  1834.         }
  1835.       } catch(e) { relatedFilesDir = null; }
  1836.     }
  1837.  
  1838.     var destinationLocation;
  1839.     if (tempLocalFile)
  1840.       destinationLocation = tempLocalFile;
  1841.     else
  1842.       destinationLocation = docURI;
  1843.  
  1844.     success = OutputFileWithPersistAPI(editorDoc, destinationLocation, relatedFilesDir, aMimeType);
  1845.   }
  1846.   catch (e)
  1847.   {
  1848.     success = false;
  1849.   }
  1850.  
  1851.   if (success)
  1852.   {
  1853.     try {
  1854.       if (doUpdateURI)
  1855.       {
  1856.          // If a local file, we must create a new uri from nsILocalFile
  1857.         if (tempLocalFile)
  1858.           docURI = GetFileProtocolHandler().newFileURI(tempLocalFile);
  1859.  
  1860.         // We need to set new document uri before notifying listeners
  1861.         SetDocumentURI(docURI);
  1862.       }
  1863.  
  1864.       // Update window title to show possibly different filename
  1865.       // This also covers problem that after undoing a title change,
  1866.       //   window title loses the extra [filename] part that this adds
  1867.       UpdateWindowTitle();
  1868.  
  1869.       if (!aSaveCopy)
  1870.         editor.resetModificationCount();
  1871.       // this should cause notification to listeners that document has changed
  1872.  
  1873.       // Set UI based on whether we're editing a remote or local url
  1874.       SetSaveAndPublishUI(urlstring);
  1875.     } catch (e) {}
  1876.   }
  1877.   else
  1878.   {
  1879.     var saveDocStr = GetString("SaveDocument");
  1880.     var failedStr = GetString("SaveFileFailed");
  1881.     AlertWithTitle(saveDocStr, failedStr);
  1882.   }
  1883.   return success;
  1884. }
  1885.  
  1886. function SetDocumentURI(uri)
  1887. {
  1888.   try {
  1889.     // XXX WE'LL NEED TO GET "CURRENT" CONTENT FRAME ONCE MULTIPLE EDITORS ARE ALLOWED
  1890.     GetCurrentEditorElement().docShell.setCurrentURI(uri);
  1891.   } catch (e) { dump("SetDocumentURI:\n"+e +"\n"); }
  1892. }
  1893.  
  1894.  
  1895. //-------------------------------  Publishing
  1896. var gPublishData;
  1897. var gProgressDialog;
  1898. var gCommandAfterPublishing = null;
  1899. var gRestoreDocumentSource;
  1900.  
  1901. function Publish(publishData)
  1902. {
  1903.   if (!publishData)
  1904.     return false;
  1905.  
  1906.   // Set data in global for username password requests
  1907.   //  and to do "post saving" actions after monitoring nsIWebProgressListener messages
  1908.   //  and we are sure file transfer was successful
  1909.   gPublishData = publishData;
  1910.  
  1911.   gPublishData.docURI = CreateURIFromPublishData(publishData, true);
  1912.   if (!gPublishData.docURI)
  1913.   {
  1914.     AlertWithTitle(GetString("Publish"), GetString("PublishFailed"));
  1915.     return false;
  1916.   }
  1917.  
  1918.   if (gPublishData.publishOtherFiles)
  1919.     gPublishData.otherFilesURI = CreateURIFromPublishData(publishData, false);
  1920.   else
  1921.     gPublishData.otherFilesURI = null;
  1922.  
  1923.   if (gShowDebugOutputStateChange)
  1924.   {
  1925.     dump("\n *** publishData: PublishUrl="+publishData.publishUrl+", BrowseUrl="+publishData.browseUrl+
  1926.       ", Username="+publishData.username+", Dir="+publishData.docDir+
  1927.       ", Filename="+publishData.filename+"\n");
  1928.     dump(" * gPublishData.docURI.spec w/o pass="+StripPassword(gPublishData.docURI.spec)+", PublishOtherFiles="+gPublishData.publishOtherFiles+"\n");
  1929.   }
  1930.  
  1931.   // XXX Missing username will make FTP fail 
  1932.   // and it won't call us for prompt dialog (bug 132320)
  1933.   // (It does prompt if just password is missing)
  1934.   // So we should do the prompt ourselves before trying to publish
  1935.   if (GetScheme(publishData.publishUrl) == "ftp" && !publishData.username)
  1936.   {
  1937.     var message = GetString("PromptFTPUsernamePassword").replace(/%host%/, GetHost(publishData.publishUrl));
  1938.     var savePWobj = {value:publishData.savePassword};
  1939.     var userObj = {value:publishData.username};
  1940.     var pwObj = {value:publishData.password};
  1941.     if (!PromptUsernameAndPassword(GetString("Prompt"), message, savePWobj, userObj, pwObj))
  1942.       return false; // User canceled out of dialog
  1943.  
  1944.     // Reset data in URI objects
  1945.     gPublishData.docURI.username = publishData.username;
  1946.     gPublishData.docURI.password = publishData.password;
  1947.  
  1948.     if (gPublishData.otherFilesURI)
  1949.     {
  1950.       gPublishData.otherFilesURI.username = publishData.username;
  1951.       gPublishData.otherFilesURI.password = publishData.password;
  1952.     }
  1953.   }
  1954.  
  1955.   try {
  1956.     // We launch dialog as a dependent 
  1957.     // Don't allow editing document!
  1958.     SetDocumentEditable(false);
  1959.  
  1960.     // Start progress monitoring
  1961.     gProgressDialog =
  1962.       window.openDialog("chrome://editor/content/EditorPublishProgress.xul", "_blank",
  1963.                         "chrome,dependent,titlebar", gPublishData, gPersistObj);
  1964.  
  1965.   } catch (e) {}
  1966.  
  1967.   // Network transfer is often too quick for the progress dialog to be initialized
  1968.   //  and we can completely miss messages for quickly-terminated bad URLs,
  1969.   //  so we can't call OutputFileWithPersistAPI right away.
  1970.   // StartPublishing() is called at the end of the dialog's onload method
  1971.   return true;
  1972. }
  1973.  
  1974. function StartPublishing()
  1975. {
  1976.   var editor = GetCurrentEditor();
  1977.   if (editor && gPublishData && gPublishData.docURI && gProgressDialog)
  1978.   {
  1979.     gRestoreDocumentSource = null;
  1980.  
  1981.     // Save backup document since nsIWebBrowserPersist changes image src urls
  1982.     // but we only need to do this if publishing images and other related files
  1983.     if (gPublishData.otherFilesURI)
  1984.     {
  1985.       try {
  1986.         // (256 = Output encoded entities)
  1987.         gRestoreDocumentSource = 
  1988.           editor.outputToString(editor.contentsMIMEType, 256);
  1989.       } catch (e) {}
  1990.     }
  1991.  
  1992.     OutputFileWithPersistAPI(editor.document, 
  1993.                              gPublishData.docURI, gPublishData.otherFilesURI, 
  1994.                              editor.contentsMIMEType);
  1995.     return gPersistObj;
  1996.   }
  1997.   return null;
  1998. }
  1999.  
  2000. function CancelPublishing()
  2001. {
  2002.   try {
  2003.     gPersistObj.cancelSave();
  2004.     gProgressDialog.SetProgressStatusCancel();
  2005.   } catch (e) {}
  2006.  
  2007.   // If canceling publishing do not do any commands after this    
  2008.   gCommandAfterPublishing = null;
  2009.  
  2010.   if (gProgressDialog)
  2011.   {
  2012.     // Close Progress dialog 
  2013.     // (this will call FinishPublishing())
  2014.     gProgressDialog.CloseDialog();
  2015.   }
  2016.   else
  2017.     FinishPublishing();
  2018. }
  2019.  
  2020. function FinishPublishing()
  2021. {
  2022.   SetDocumentEditable(true);
  2023.   gProgressDialog = null;
  2024.   gPublishData = null;
  2025.   gRestoreDocumentSource = null;
  2026.  
  2027.   if (gCommandAfterPublishing)
  2028.   {
  2029.     // Be sure to null out the global now incase of trouble when executing command
  2030.     var command = gCommandAfterPublishing;
  2031.     gCommandAfterPublishing = null;
  2032.     goDoCommand(command);
  2033.   }
  2034. }
  2035.  
  2036. // Create a nsIURI object filled in with all required publishing info
  2037. function CreateURIFromPublishData(publishData, doDocUri)
  2038. {
  2039.   if (!publishData || !publishData.publishUrl)
  2040.     return null;
  2041.  
  2042.   var URI;
  2043.   try {
  2044.     var spec = publishData.publishUrl;
  2045.     if (doDocUri)
  2046.       spec += FormatDirForPublishing(publishData.docDir) + publishData.filename; 
  2047.     else
  2048.       spec += FormatDirForPublishing(publishData.otherDir);
  2049.  
  2050.     var ioService = GetIOService();
  2051.     URI = ioService.newURI(spec, GetCurrentEditor().documentCharacterSet, null);
  2052.  
  2053.     if (publishData.username)
  2054.       URI.username = publishData.username;
  2055.     if (publishData.password)
  2056.       URI.password = publishData.password;
  2057.   }
  2058.   catch (e) {}
  2059.  
  2060.   return URI;
  2061. }
  2062.  
  2063. // Resolve the correct "http:" document URL when publishing via ftp
  2064. function GetDocUrlFromPublishData(publishData)
  2065. {
  2066.   if (!publishData || !publishData.filename || !publishData.publishUrl)
  2067.     return "";
  2068.  
  2069.   // If user was previously editing an "ftp" url, then keep that as the new scheme
  2070.   var url;
  2071.   var docScheme = GetScheme(GetDocumentUrl());
  2072.  
  2073.   // Always use the "HTTP" address if available
  2074.   // XXX Should we do some more validation here for bad urls???
  2075.   // Let's at least check for a scheme!
  2076.   if (!GetScheme(publishData.browseUrl))
  2077.     url = publishData.publishUrl;
  2078.   else
  2079.     url = publishData.browseUrl;
  2080.  
  2081.   url += FormatDirForPublishing(publishData.docDir) + publishData.filename;
  2082.  
  2083.   if (GetScheme(url) == "ftp")
  2084.     url = InsertUsernameIntoUrl(url, publishData.username);
  2085.  
  2086.   return url;
  2087. }
  2088.  
  2089. function SetSaveAndPublishUI(urlstring)
  2090. {
  2091.   // Be sure enabled state of toolbar buttons are correct
  2092.   goUpdateCommand("cmd_save");
  2093.   goUpdateCommand("cmd_publish");
  2094. }
  2095.  
  2096. function SetDocumentEditable(isDocEditable)
  2097. {
  2098.   var editor = GetCurrentEditor();
  2099.   if (editor && editor.document)
  2100.   {
  2101.     try {
  2102.       var flags = editor.flags;
  2103.       editor.flags = isDocEditable ?  
  2104.             flags &= ~nsIPlaintextEditor.eEditorReadonlyMask :
  2105.             flags | nsIPlaintextEditor.eEditorReadonlyMask;
  2106.     } catch(e) {}
  2107.  
  2108.     // update all commands
  2109.     window.updateCommands("create");
  2110.   }  
  2111. }
  2112.  
  2113. // ****** end of save / publish **********//
  2114.  
  2115. //-----------------------------------------------------------------------------------
  2116. var nsPublishSettingsCommand =
  2117. {
  2118.   isCommandEnabled: function(aCommand, dummy)
  2119.   {
  2120.     return (IsDocumentEditable());
  2121.   },
  2122.  
  2123.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2124.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2125.  
  2126.   doCommand: function(aCommand)
  2127.   {
  2128.     if (GetCurrentEditor())
  2129.     {
  2130.       // Launch Publish Settings dialog
  2131.  
  2132.       window.ok = window.openDialog("chrome://editor/content/EditorPublishSettings.xul","_blank", "chrome,close,titlebar,modal", "");
  2133.       window.content.focus();
  2134.       return window.ok;
  2135.     }
  2136.     return false;
  2137.   }
  2138. }
  2139.  
  2140. //-----------------------------------------------------------------------------------
  2141. var nsRevertCommand =
  2142. {
  2143.   isCommandEnabled: function(aCommand, dummy)
  2144.   {
  2145.     return (IsDocumentEditable() &&
  2146.             IsDocumentModified() &&
  2147.             !IsUrlAboutBlank(GetDocumentUrl()));
  2148.   },
  2149.  
  2150.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2151.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2152.  
  2153.   doCommand: function(aCommand)
  2154.   {
  2155.     // Confirm with the user to abandon current changes
  2156.     var promptService = GetPromptService();
  2157.     if (promptService)
  2158.     {
  2159.       // Put the page title in the message string
  2160.       var title = GetDocumentTitle();
  2161.       if (!title)
  2162.         title = GetString("untitled");
  2163.  
  2164.       var msg = GetString("AbandonChanges").replace(/%title%/,title);
  2165.  
  2166.       var result = promptService.confirmEx(window, GetString("RevertCaption"), msg,
  2167.                                 (promptService.BUTTON_TITLE_REVERT * promptService.BUTTON_POS_0) +
  2168.                                 (promptService.BUTTON_TITLE_CANCEL * promptService.BUTTON_POS_1),
  2169.                                 null, null, null, null, {value:0});
  2170.  
  2171.       // Reload page if first button (Revert) was pressed
  2172.       if(result == 0)
  2173.       {
  2174.         CancelHTMLSource();
  2175.         EditorLoadUrl(GetDocumentUrl());
  2176.       }
  2177.     }
  2178.   }
  2179. };
  2180.  
  2181. //-----------------------------------------------------------------------------------
  2182. var nsCloseCommand =
  2183. {
  2184.   isCommandEnabled: function(aCommand, dummy)
  2185.   {
  2186.     return GetCurrentEditor() != null;
  2187.   },
  2188.   
  2189.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2190.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2191.  
  2192.   doCommand: function(aCommand)
  2193.   {
  2194.     CloseCurrentTab(true);
  2195.   }
  2196. };
  2197.  
  2198. //-----------------------------------------------------------------------------------
  2199. var nsCloseAllCommand =
  2200. {
  2201.   isCommandEnabled: function(aCommand, dummy)
  2202.   {
  2203.     return GetCurrentEditor() != null;
  2204.   },
  2205.   
  2206.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2207.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2208.  
  2209.   doCommand: function(aCommand)
  2210.   {
  2211.     CloseWindow(true);
  2212.   }
  2213. };
  2214.  
  2215. function CloseCurrentTab(aDontLeaveTabeditorEmpty)
  2216. {
  2217.   var handled = CheckAndSaveDocument("cmd_close", true);
  2218.   if (handled) 
  2219.   {
  2220.     var tabeditor = document.getElementById("tabeditor");
  2221.     var tab    = tabeditor.selectedTab;
  2222.     var index  = tabeditor.selectedIndex;
  2223.     var n      = tabeditor.mTabpanels.childNodes.length;
  2224.     var editor = tabeditor.selectedPanel;
  2225.  
  2226.     var newIndex = Math.min(index, n - 2);
  2227.  
  2228.     editor.parentNode.removeChild(editor);
  2229.     tab.parentNode.removeChild(tab);
  2230.     if (newIndex >= 0)
  2231.       tabeditor.selectedIndex = newIndex;
  2232.     else if (aDontLeaveTabeditorEmpty)
  2233.       tabeditor.newBlankTab();
  2234.   }
  2235.   return handled;
  2236. }
  2237.  
  2238. function CloseTab(tab)
  2239. {
  2240.   if (!tab)
  2241.     return;
  2242.  
  2243.   var tabeditor = document.getElementById("tabeditor");
  2244.   var currentTab = tabeditor.selectedTab;
  2245.   var currentIndex = tabeditor.selectedIndex;
  2246.   var n  = tabeditor.mTabpanels.childNodes.length;
  2247.  
  2248.   var newIndex = Math.min(currentIndex, n - 2);
  2249.  
  2250.   if (tab != currentTab)
  2251.     tabeditor.selectedTab = tab;
  2252.  
  2253.   var handled = CheckAndSaveDocument("cmd_close", true);
  2254.   if (handled) 
  2255.   {
  2256.     var editor = tabeditor.selectedPanel;
  2257.     editor.parentNode.removeChild(editor);
  2258.     tab.parentNode.removeChild(tab);
  2259.  
  2260.     if (newIndex >= 0)
  2261.     {
  2262.       if (tab != currentTab)
  2263.         tabeditor.selectedTab = currentTab;
  2264.       else
  2265.         tabeditor.selectedIndex = newIndex;
  2266.     }
  2267.     else if (aDontLeaveTabeditorEmpty)
  2268.       tabeditor.newBlankTab();
  2269.   }
  2270. }
  2271.  
  2272. function RemoveAllTabsBut(tabToKeep)
  2273. {
  2274.   if (!tabToKeep)
  2275.     return;
  2276.  
  2277.   var tabeditor = document.getElementById("tabeditor");
  2278.   var tabs      = tabeditor.mTabs;
  2279.   var tab       = tabs.lastChild;
  2280.   while (tab)
  2281.   {
  2282.     var tmp = tab.previousSibling;
  2283.     if (tabToKeep != tab)
  2284.     {
  2285.       CloseTab(tab);
  2286.     }
  2287.     tab = tmp;
  2288.   }
  2289. }
  2290.  
  2291. function CloseWindow(aCloseOnlyTabs)
  2292. {
  2293.   // Check to make sure document is saved. "true" means allow "Don't Save" button,
  2294.   //   so user can choose to close without saving
  2295.   var tabeditor = document.getElementById("tabeditor");
  2296.   var i, n = tabeditor.mTabs.childNodes.length;
  2297.   tabeditor.selectedIndex = 0;
  2298.   for (i=0; i<n; i++)
  2299.   {
  2300.     if (!CloseCurrentTab(aCloseOnlyTabs))
  2301.       return false;
  2302.   }
  2303.  
  2304.   if (aCloseOnlyTabs)
  2305.     return true;
  2306.  
  2307.   if (window.InsertCharWindow)
  2308.     SwitchInsertCharToAnotherEditorOrClose();
  2309.  
  2310.   try {
  2311.     var basewin = window.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
  2312.                     .getInterface(Components.interfaces.nsIWebNavigation)
  2313.                     .QueryInterface(Components.interfaces.nsIDocShellTreeItem)
  2314.                     .treeOwner
  2315.                     .QueryInterface(Components.interfaces.nsIInterfaceRequestor)
  2316.                     .getInterface(Components.interfaces.nsIBaseWindow);
  2317.     basewin.destroy();
  2318.   } catch (e) {}
  2319.   return true;
  2320. }
  2321.  
  2322. //-----------------------------------------------------------------------------------
  2323. var nsOpenRemoteCommand =
  2324. {
  2325.   isCommandEnabled: function(aCommand, dummy)
  2326.   {
  2327.     return true;    // we can always do this
  2328.   },
  2329.  
  2330.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2331.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2332.  
  2333.   doCommand: function(aCommand)
  2334.   {
  2335.       /* The last parameter is the current browser window.
  2336.          Use 0 and the default checkbox will be to load into an editor
  2337.          and loading into existing browser option is removed
  2338.        */
  2339.       window.openDialog( "chrome://communicator/content/openLocation.xul", "_blank", "chrome,modal,titlebar", 0);
  2340.     window.content.focus();
  2341.   }
  2342. };
  2343.  
  2344. //-----------------------------------------------------------------------------------
  2345. var nsNewEditedObjectCommand =
  2346. {
  2347.   isCommandEnabled: function(aCommand, dummy)
  2348.   {
  2349.     return true;    // we can always do this
  2350.   },
  2351.  
  2352.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2353.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2354.  
  2355.   doCommand: function(aCommand)
  2356.   {
  2357.       /* The last parameter is the current browser window.
  2358.          Use 0 and the default checkbox will be to load into an editor
  2359.          and loading into existing browser option is removed
  2360.        */
  2361.       window.openDialog( "chrome://communicator/content/newDocument.xul", "_blank", "chrome,modal,titlebar", 0);
  2362.     window.content.focus();
  2363.   }
  2364. };
  2365.  
  2366. //-----------------------------------------------------------------------------------
  2367. var nsPreviewCommand =
  2368. {
  2369.   isCommandEnabled: function(aCommand, dummy)
  2370.   {
  2371.     return (IsDocumentEditable() && 
  2372.             IsHTMLEditor() && 
  2373.             (DocumentHasBeenSaved() || IsDocumentModified()));
  2374.   },
  2375.  
  2376.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2377.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2378.  
  2379.   doCommand: function(aCommand)
  2380.   {
  2381.       // Don't continue if user canceled during prompt for saving
  2382.     // DocumentHasBeenSaved will test if we have a URL and suppress "Don't Save" button if not
  2383.     if (!CheckAndSaveDocument("cmd_preview", DocumentHasBeenSaved()))
  2384.         return;
  2385.  
  2386.     // Check if we saved again just in case?
  2387.       if (DocumentHasBeenSaved())
  2388.     {
  2389.       loadExternalURL(GetDocumentUrl());
  2390.       
  2391. /*
  2392.       var browser;
  2393.       try {
  2394.         // Find a browser with this URL
  2395.         var windowManager = Components.classes["@mozilla.org/appshell/window-mediator;1"].getService();
  2396.         var windowManagerInterface = windowManager.QueryInterface(Components.interfaces.nsIWindowMediator);
  2397.         var enumerator = windowManagerInterface.getEnumerator("navigator:browser");
  2398.  
  2399.         var documentURI = GetDocumentUrl();
  2400.         while ( enumerator.hasMoreElements() )
  2401.         {
  2402.           browser = enumerator.getNext().QueryInterface(Components.interfaces.nsIDOMWindowInternal);
  2403.           if ( browser && (documentURI == browser.getBrowser().currentURI.spec))
  2404.             break;
  2405.  
  2406.           browser = null;
  2407.         }
  2408.       }
  2409.       catch (ex) {}
  2410.  
  2411.       // If none found, open a new browser
  2412.       if (!browser)
  2413.       {
  2414.         browser = window.openDialog(getBrowserURL(), "_blank", "chrome,all,dialog=no", documentURI);
  2415.       }
  2416.       else
  2417.       {
  2418.         try {
  2419.           browser.BrowserReloadSkipCache();
  2420.           browser.focus();
  2421.         } catch (ex) {}
  2422.       }
  2423. */
  2424.     }
  2425.   }
  2426. };
  2427.  
  2428. //-----------------------------------------------------------------------------------
  2429. var nsSendPageCommand =
  2430. {
  2431.   isCommandEnabled: function(aCommand, dummy)
  2432.   {
  2433.     return (IsDocumentEditable() &&
  2434.             (DocumentHasBeenSaved() || IsDocumentModified()));
  2435.   },
  2436.  
  2437.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2438.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2439.  
  2440.   doCommand: function(aCommand)
  2441.   {
  2442.     // Don't continue if user canceled during prompt for saving
  2443.     // DocumentHasBeenSaved will test if we have a URL and suppress "Don't Save" button if not
  2444.     if (!CheckAndSaveDocument("cmd_editSendPage", DocumentHasBeenSaved()))
  2445.         return;
  2446.  
  2447.     // Check if we saved again just in case?
  2448.     if (DocumentHasBeenSaved())
  2449.     {
  2450.       // Launch Messenger Composer window with current page as contents
  2451.       try
  2452.       {
  2453.         openComposeWindow(GetDocumentUrl(), GetDocumentTitle());        
  2454.       } catch (ex) { dump("Cannot Send Page: " + ex + "\n"); }
  2455.     }
  2456.   }
  2457. };
  2458.  
  2459. //-----------------------------------------------------------------------------------
  2460. var nsPrintCommand =
  2461. {
  2462.   isCommandEnabled: function(aCommand, dummy)
  2463.   {
  2464.     return true;    // we can always do this
  2465.   },
  2466.  
  2467.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2468.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2469.  
  2470.   doCommand: function(aCommand)
  2471.   {
  2472.     // In editor.js
  2473.     FinishHTMLSource();
  2474.     try {
  2475.       NSPrint();
  2476.     } catch (e) {}
  2477.   }
  2478. };
  2479.  
  2480. //-----------------------------------------------------------------------------------
  2481. var nsPrintSetupCommand =
  2482. {
  2483.   isCommandEnabled: function(aCommand, dummy)
  2484.   {
  2485.     return true;    // we can always do this
  2486.   },
  2487.  
  2488.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2489.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2490.  
  2491.   doCommand: function(aCommand)
  2492.   {
  2493.     // In editor.js
  2494.     FinishHTMLSource();
  2495.     NSPrintSetup();
  2496.   }
  2497. };
  2498.  
  2499. //-----------------------------------------------------------------------------------
  2500. var nsQuitCommand =
  2501. {
  2502.   isCommandEnabled: function(aCommand, dummy)
  2503.   {
  2504.     return true;    // we can always do this
  2505.   },
  2506.  
  2507.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2508.   doCommandParams: function(aCommand, aParams, aRefCon) {}
  2509.  
  2510.   /* The doCommand is not used, since cmd_quit's oncommand="goQuitApplication()" in platformCommunicatorOverlay.xul
  2511.   doCommand: function(aCommand)
  2512.   {
  2513.     // In editor.js
  2514.     FinishHTMLSource();
  2515.     goQuitApplication();
  2516.   }
  2517.   */
  2518. };
  2519.  
  2520. //-----------------------------------------------------------------------------------
  2521. var nsFindCommand =
  2522. {
  2523.   isCommandEnabled: function(aCommand, editorElement)
  2524.   {
  2525.     return editorElement.getEditor(editorElement.contentWindow) != null;
  2526.   },
  2527.  
  2528.   getCommandStateParams: function(aCommand, aParams, editorElement) {},
  2529.   doCommandParams: function(aCommand, aParams, editorElement) {},
  2530.  
  2531.   doCommand: function(aCommand, editorElement)
  2532.   {
  2533.     try {
  2534.       window.openDialog("chrome://editor/content/EdReplace.xul", "_blank",
  2535.                         "chrome,modal,titlebar", editorElement);
  2536.     }
  2537.     catch(ex) {
  2538.       dump("*** Exception: couldn't open Replace Dialog\n");
  2539.     }
  2540.     //window.content.focus();
  2541.   }
  2542. };
  2543.  
  2544. //-----------------------------------------------------------------------------------
  2545. var nsFindAgainCommand =
  2546. {
  2547.   isCommandEnabled: function(aCommand, editorElement)
  2548.   {
  2549.     // we can only do this if the search pattern is non-empty. Not sure how
  2550.     // to get that from here
  2551.     return editorElement.getEditor(editorElement.contentWindow) != null;
  2552.   },
  2553.  
  2554.   getCommandStateParams: function(aCommand, aParams, editorElement) {},
  2555.   doCommandParams: function(aCommand, aParams, editorElement) {},
  2556.  
  2557.   doCommand: function(aCommand, editorElement)
  2558.   {
  2559.     try {
  2560.       var findPrev = aCommand == "cmd_findPrev";
  2561.       var findInst = editorElement.webBrowserFind;
  2562.       var findService = Components.classes["@mozilla.org/find/find_service;1"]
  2563.                                   .getService(Components.interfaces.nsIFindService);
  2564.       findInst.findBackwards = findService.findBackwards ^ findPrev;
  2565.       findInst.findNext();
  2566.       // reset to what it was in dialog, otherwise dialog setting can get reversed
  2567.       findInst.findBackwards = findService.findBackwards; 
  2568.     }
  2569.     catch (ex) {}
  2570.   }
  2571. };
  2572.  
  2573. //-----------------------------------------------------------------------------------
  2574. var nsRewrapCommand =
  2575. {
  2576.   isCommandEnabled: function(aCommand, dummy)
  2577.   {
  2578.     return (IsDocumentEditable() && !IsInHTMLSourceMode() &&
  2579.             GetCurrentEditor() instanceof Components.interfaces.nsIEditorMailSupport);
  2580.   },
  2581.  
  2582.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2583.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2584.  
  2585.   doCommand: function(aCommand)
  2586.   {
  2587.     GetCurrentEditor().QueryInterface(Components.interfaces.nsIEditorMailSupport).rewrap(false);
  2588.   }
  2589. };
  2590.  
  2591. //-----------------------------------------------------------------------------------
  2592. var nsSpellingCommand =
  2593. {
  2594.   isCommandEnabled: function(aCommand, dummy)
  2595.   {
  2596.     return (IsDocumentEditable() && 
  2597.             !IsInHTMLSourceMode() && IsSpellCheckerInstalled());
  2598.   },
  2599.  
  2600.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2601.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2602.  
  2603.   doCommand: function(aCommand)
  2604.   {
  2605.     window.cancelSendMessage = false;
  2606.     try {
  2607.       var skipBlockQuotes = (window.document.firstChild.getAttribute("windowtype") == "msgcompose");
  2608.       window.openDialog("chrome://editor/content/EdSpellCheck.xul", "_blank",
  2609.               "chrome,close,titlebar,modal", false, skipBlockQuotes, true);
  2610.     }
  2611.     catch(ex) {}
  2612.     window.content.focus();
  2613.   }
  2614. };
  2615.  
  2616. //-----------------------------------------------------------------------------------
  2617. var nsCleanupCommand =
  2618. {
  2619.   isCommandEnabled: function(aCommand, dummy)
  2620.   {
  2621.     return (IsDocumentEditable() && 
  2622.             !IsInHTMLSourceMode());
  2623.   },
  2624.  
  2625.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2626.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2627.  
  2628.   doCommand: function(aCommand)
  2629.   {
  2630.     try {
  2631.       window.openDialog("chrome://editor/content/MarkupCleaner.xul", "_blank",
  2632.               "chrome,close,titlebar,modal");
  2633.     }
  2634.     catch(ex) {}
  2635.     window.content.focus();
  2636.   }
  2637. };
  2638.  
  2639. //-----------------------------------------------------------------------------------
  2640. // Validate using http://validator.w3.org/file-upload.html
  2641. var URL2Validate;
  2642. var nsValidateCommand =
  2643. {
  2644.   isCommandEnabled: function(aCommand, dummy)
  2645.   {
  2646.     return GetCurrentEditor() != null;
  2647.   },
  2648.  
  2649.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2650.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2651.  
  2652.   doCommand: function(aCommand)
  2653.   {
  2654.     // If the document hasn't been modified,
  2655.     // then just validate the current url.
  2656.     if (IsDocumentModified() || IsHTMLSourceChanged())
  2657.     {
  2658.       if (!CheckAndSaveDocument("cmd_validate", false))
  2659.         return;
  2660.  
  2661.       // Check if we saved again just in case?
  2662.       if (!DocumentHasBeenSaved())    // user hit cancel?
  2663.         return;
  2664.     }
  2665.  
  2666.     URL2Validate = GetDocumentUrl();
  2667.     // See if it's a file:
  2668.     var ifile;
  2669.     try {
  2670.       var fileHandler = GetFileProtocolHandler();
  2671.       ifile = fileHandler.getFileFromURLSpec(URL2Validate);
  2672.       // nsIFile throws an exception if it's not a file url
  2673.     } catch (e) { ifile = null; }
  2674.     if (ifile)
  2675.     {
  2676.       window.openDialog("chrome://editor/content/Validator.xul", "_blank", "chrome,close,titlebar,resizable=yes", URL2Validate, true);
  2677.     }
  2678.     else
  2679.     {
  2680.       window.openDialog("chrome://editor/content/Validator.xul", "_blank", "chrome,close,titlebar,resizable=yes", URL2Validate, false);
  2681.     }
  2682.   },
  2683.   validateFilePageLoaded: function(event)
  2684.   {
  2685.     event.target.forms[0].uploaded_file.value = URL2Validate;
  2686.   }
  2687. };
  2688.  
  2689. var nsCheckLinksCommand =
  2690. {
  2691.   isCommandEnabled: function(aCommand, dummy)
  2692.   {
  2693.     return (IsDocumentEditable());
  2694.   },
  2695.  
  2696.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2697.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2698.  
  2699.   doCommand: function(aCommand)
  2700.   {
  2701.     window.openDialog("chrome://editor/content/EdLinkChecker.xul","_blank", "chrome,close,titlebar,modal");
  2702.     window.content.focus();
  2703.   }
  2704. };
  2705.  
  2706. //-----------------------------------------------------------------------------------
  2707. var nsFormCommand =
  2708. {
  2709.   isCommandEnabled: function(aCommand, dummy)
  2710.   {
  2711.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2712.   },
  2713.  
  2714.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2715.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2716.  
  2717.   doCommand: function(aCommand)
  2718.   {
  2719.     window.openDialog("chrome://editor/content/EdFormProps.xul", "_blank", "chrome,close,titlebar,modal");
  2720.     window.content.focus();
  2721.   }
  2722. };
  2723.  
  2724. //-----------------------------------------------------------------------------------
  2725. var nsInputTagCommand =
  2726. {
  2727.   isCommandEnabled: function(aCommand, dummy)
  2728.   {
  2729.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2730.   },
  2731.  
  2732.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2733.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2734.  
  2735.   doCommand: function(aCommand)
  2736.   {
  2737.     window.openDialog("chrome://editor/content/EdInputProps.xul", "_blank", "chrome,close,titlebar,modal");
  2738.     window.content.focus();
  2739.   }
  2740. };
  2741.  
  2742. //-----------------------------------------------------------------------------------
  2743. var nsInputImageCommand =
  2744. {
  2745.   isCommandEnabled: function(aCommand, dummy)
  2746.   {
  2747.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2748.   },
  2749.  
  2750.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2751.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2752.  
  2753.   doCommand: function(aCommand)
  2754.   {
  2755.     window.openDialog("chrome://editor/content/EdInputImage.xul", "_blank", "chrome,close,titlebar,modal");
  2756.     window.content.focus();
  2757.   }
  2758. };
  2759.  
  2760. //-----------------------------------------------------------------------------------
  2761. var nsTextAreaCommand =
  2762. {
  2763.   isCommandEnabled: function(aCommand, dummy)
  2764.   {
  2765.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2766.   },
  2767.  
  2768.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2769.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2770.  
  2771.   doCommand: function(aCommand)
  2772.   {
  2773.     window.openDialog("chrome://editor/content/EdTextAreaProps.xul", "_blank", "chrome,close,titlebar,modal");
  2774.     window.content.focus();
  2775.   }
  2776. };
  2777.  
  2778. //-----------------------------------------------------------------------------------
  2779. var nsSelectCommand =
  2780. {
  2781.   isCommandEnabled: function(aCommand, dummy)
  2782.   {
  2783.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2784.   },
  2785.  
  2786.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2787.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2788.  
  2789.   doCommand: function(aCommand)
  2790.   {
  2791.     window.openDialog("chrome://editor/content/EdSelectProps.xul", "_blank", "chrome,close,titlebar,modal");
  2792.     window.content.focus();
  2793.   }
  2794. };
  2795.  
  2796. //-----------------------------------------------------------------------------------
  2797. var nsButtonCommand =
  2798. {
  2799.   isCommandEnabled: function(aCommand, dummy)
  2800.   {
  2801.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2802.   },
  2803.  
  2804.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2805.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2806.  
  2807.   doCommand: function(aCommand)
  2808.   {
  2809.     window.openDialog("chrome://editor/content/EdButtonProps.xul", "_blank", "chrome,close,titlebar,modal");
  2810.     window.content.focus();
  2811.   }
  2812. };
  2813.  
  2814. //-----------------------------------------------------------------------------------
  2815. var nsLabelCommand =
  2816. {
  2817.   isCommandEnabled: function(aCommand, dummy)
  2818.   {
  2819.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2820.   },
  2821.  
  2822.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2823.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2824.  
  2825.   doCommand: function(aCommand)
  2826.   {
  2827.     var tagName = "label";
  2828.     try {
  2829.       var editor = GetCurrentEditor();
  2830.       // Find selected label or if start/end of selection is in label 
  2831.       var labelElement = editor.getSelectedElement(tagName);
  2832.       if (!labelElement)
  2833.         labelElement = editor.getElementOrParentByTagName(tagName, editor.selection.anchorNode);
  2834.       if (!labelElement)
  2835.         labelElement = editor.getElementOrParentByTagName(tagName, editor.selection.focusNode);
  2836.       if (labelElement) {
  2837.         // We only open the dialog for an existing label
  2838.         window.openDialog("chrome://editor/content/EdLabelProps.xul", "_blank", "chrome,close,titlebar,modal", labelElement);
  2839.         window.content.focus();
  2840.       } else {
  2841.         EditorSetTextProperty(tagName, "", "");
  2842.       }
  2843.     } catch (e) {}
  2844.   }
  2845. };
  2846.  
  2847. //-----------------------------------------------------------------------------------
  2848. var nsFieldSetCommand =
  2849. {
  2850.   isCommandEnabled: function(aCommand, dummy)
  2851.   {
  2852.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2853.   },
  2854.  
  2855.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2856.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2857.  
  2858.   doCommand: function(aCommand)
  2859.   {
  2860.     window.openDialog("chrome://editor/content/EdFieldSetProps.xul", "_blank", "chrome,close,titlebar,modal");
  2861.     window.content.focus();
  2862.   }
  2863. };
  2864.  
  2865. //-----------------------------------------------------------------------------------
  2866. var nsIsIndexCommand =
  2867. {
  2868.   isCommandEnabled: function(aCommand, dummy)
  2869.   {
  2870.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2871.   },
  2872.  
  2873.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2874.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2875.  
  2876.   doCommand: function(aCommand)
  2877.   {
  2878.     try {
  2879.       var editor = GetCurrentEditor();
  2880.       var isindexElement = editor.createElementWithDefaults("isindex");
  2881.       isindexElement.setAttribute("prompt", editor.outputToString("text/plain", 1)); // OutputSelectionOnly
  2882.       editor.insertElementAtSelection(isindexElement, true);
  2883.     } catch (e) {}
  2884.   }
  2885. };
  2886.  
  2887. //-----------------------------------------------------------------------------------
  2888. var nsImageCommand =
  2889. {
  2890.   isCommandEnabled: function(aCommand, dummy)
  2891.   {
  2892.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2893.   },
  2894.  
  2895.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2896.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2897.  
  2898.   doCommand: function(aCommand)
  2899.   {
  2900.     window.openDialog("chrome://editor/content/EdImageProps.xul","_blank", "chrome,close,titlebar,modal");
  2901.     window.content.focus();
  2902.   }
  2903. };
  2904.  
  2905. //-----------------------------------------------------------------------------------
  2906. var nsHLineCommand =
  2907. {
  2908.   isCommandEnabled: function(aCommand, dummy)
  2909.   {
  2910.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2911.   },
  2912.  
  2913.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2914.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2915.  
  2916.   doCommand: function(aCommand)
  2917.   {
  2918.     // Inserting an HLine is different in that we don't use properties dialog
  2919.     //  unless we are editing an existing line's attributes
  2920.     //  We get the last-used attributes from the prefs and insert immediately
  2921.  
  2922.     var tagName = "hr";
  2923.     var editor = GetCurrentEditor();
  2924.       
  2925.     var hLine;
  2926.     try {
  2927.       hLine = editor.getSelectedElement(tagName);
  2928.     } catch (e) {return;}
  2929.  
  2930.     if (hLine)
  2931.     {
  2932.       // We only open the dialog for an existing HRule
  2933.       window.openDialog("chrome://editor/content/EdHLineProps.xul", "_blank", "chrome,close,titlebar,modal");
  2934.       window.content.focus();
  2935.     } 
  2936.     else
  2937.     {
  2938.       try {
  2939.         hLine = editor.createElementWithDefaults(tagName);
  2940.  
  2941.         // We change the default attributes to those saved in the user prefs
  2942.         var prefs = GetPrefs();
  2943.         var align = prefs.getIntPref("editor.hrule.align");
  2944.         if (align == 0)
  2945.           editor.setAttributeOrEquivalent(hLine, "align", "left", true);
  2946.         else if (align == 2)
  2947.           editor.setAttributeOrEquivalent(hLine, "align", "right", true);
  2948.  
  2949.         //Note: Default is center (don't write attribute)
  2950.   
  2951.         var width = prefs.getIntPref("editor.hrule.width");
  2952.         var percent = prefs.getBoolPref("editor.hrule.width_percent");
  2953.         if (percent)
  2954.           width = width +"%";
  2955.  
  2956.         editor.setAttributeOrEquivalent(hLine, "width", width, true);
  2957.  
  2958.         var height = prefs.getIntPref("editor.hrule.height");
  2959.         editor.setAttributeOrEquivalent(hLine, "size", String(height), true);
  2960.  
  2961.         var shading = prefs.getBoolPref("editor.hrule.shading");
  2962.         if (shading)
  2963.           hLine.removeAttribute("noshade");
  2964.         else
  2965.           hLine.setAttribute("noshade", "noshade");
  2966.  
  2967.         editor.insertElementAtSelection(hLine, true);
  2968.  
  2969.       } catch (e) {}
  2970.     }
  2971.   }
  2972. };
  2973.  
  2974. //-----------------------------------------------------------------------------------
  2975. var nsLinkCommand =
  2976. {
  2977.   isCommandEnabled: function(aCommand, dummy)
  2978.   {
  2979.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  2980.   },
  2981.  
  2982.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  2983.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  2984.  
  2985.   doCommand: function(aCommand)
  2986.   {
  2987.     // If selected element is an image, launch that dialog instead 
  2988.     // since last tab panel handles link around an image
  2989.     var element = GetObjectForProperties();
  2990.     if (element && element.nodeName.toLowerCase() == "img")
  2991.       window.openDialog("chrome://editor/content/EdImageProps.xul","_blank", "chrome,close,titlebar,modal", null, true);
  2992.     else
  2993.       window.openDialog("chrome://editor/content/EdLinkProps.xul","_blank", "chrome,close,titlebar,modal");
  2994.     window.content.focus();
  2995.   }
  2996. };
  2997.  
  2998. //-----------------------------------------------------------------------------------
  2999. var nsAnchorCommand =
  3000. {
  3001.   isCommandEnabled: function(aCommand, dummy)
  3002.   {
  3003.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3004.   },
  3005.  
  3006.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3007.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3008.  
  3009.   doCommand: function(aCommand)
  3010.   {
  3011.     window.openDialog("chrome://editor/content/EdNamedAnchorProps.xul", "_blank", "chrome,close,titlebar,modal", "");
  3012.     window.content.focus();
  3013.   }
  3014. };
  3015.  
  3016. //-----------------------------------------------------------------------------------
  3017. var nsInsertHTMLWithDialogCommand =
  3018. {
  3019.   isCommandEnabled: function(aCommand, dummy)
  3020.   {
  3021.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3022.   },
  3023.  
  3024.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3025.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3026.  
  3027.   doCommand: function(aCommand)
  3028.   {
  3029.     window.openDialog("chrome://editor/content/EdInsSrc.xul","_blank", "chrome,close,titlebar,modal,resizable", "");
  3030.     window.content.focus();
  3031.   }
  3032. };
  3033.  
  3034. //-----------------------------------------------------------------------------------
  3035. var nsInsertCharsCommand =
  3036. {
  3037.   isCommandEnabled: function(aCommand, dummy)
  3038.   {
  3039.     return (IsDocumentEditable());
  3040.   },
  3041.  
  3042.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3043.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3044.  
  3045.   doCommand: function(aCommand)
  3046.   {
  3047.     EditorFindOrCreateInsertCharWindow();
  3048.   }
  3049. };
  3050.  
  3051. //-----------------------------------------------------------------------------------
  3052. var nsInsertBreakCommand =
  3053. {
  3054.   isCommandEnabled: function(aCommand, dummy)
  3055.   {
  3056.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3057.   },
  3058.  
  3059.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3060.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3061.  
  3062.   doCommand: function(aCommand)
  3063.   {
  3064.     try {
  3065.       GetCurrentEditor().insertHTML("<br>");
  3066.     } catch (e) {}
  3067.   }
  3068. };
  3069.  
  3070. //-----------------------------------------------------------------------------------
  3071. var nsInsertBreakAllCommand =
  3072. {
  3073.   isCommandEnabled: function(aCommand, dummy)
  3074.   {
  3075.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3076.   },
  3077.  
  3078.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3079.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3080.  
  3081.   doCommand: function(aCommand)
  3082.   {
  3083.     try {
  3084.       GetCurrentEditor().insertHTML("<br clear='all'>");
  3085.     } catch (e) {}
  3086.   }
  3087. };
  3088.  
  3089. //-----------------------------------------------------------------------------------
  3090. var nsGridCommand =
  3091. {
  3092.   isCommandEnabled: function(aCommand, dummy)
  3093.   {
  3094.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3095.   },
  3096.  
  3097.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3098.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3099.  
  3100.   doCommand: function(aCommand)
  3101.   {
  3102.     window.openDialog("chrome://editor/content/EdSnapToGrid.xul","_blank", "chrome,close,titlebar,modal");
  3103.     window.content.focus();
  3104.   }
  3105. };
  3106.  
  3107.  
  3108. //-----------------------------------------------------------------------------------
  3109. var nsListPropertiesCommand =
  3110. {
  3111.   isCommandEnabled: function(aCommand, dummy)
  3112.   {
  3113.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3114.   },
  3115.  
  3116.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3117.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3118.  
  3119.   doCommand: function(aCommand)
  3120.   {
  3121.     window.openDialog("chrome://editor/content/EdListProps.xul","_blank", "chrome,close,titlebar,modal");
  3122.     window.content.focus();
  3123.   }
  3124. };
  3125.  
  3126.  
  3127. //-----------------------------------------------------------------------------------
  3128. var nsPagePropertiesCommand =
  3129. {
  3130.   isCommandEnabled: function(aCommand, dummy)
  3131.   {
  3132.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3133.   },
  3134.  
  3135.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3136.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3137.  
  3138.   doCommand: function(aCommand)
  3139.   {
  3140.     var oldTitle = GetDocumentTitle();
  3141.     window.openDialog("chrome://editor/content/EdPageProps.xul","_blank", "chrome,close,titlebar,modal", "");
  3142.  
  3143.     // Update main window title and 
  3144.     // recent menu data in prefs if doc title changed
  3145.     if (GetDocumentTitle() != oldTitle)
  3146.       UpdateWindowTitle();
  3147.  
  3148.     window.content.focus();
  3149.   }
  3150. };
  3151.  
  3152. //-----------------------------------------------------------------------------------
  3153. var nsObjectPropertiesCommand =
  3154. {
  3155.   isCommandEnabled: function(aCommand, dummy)
  3156.   {
  3157.     var isEnabled = false;
  3158.     if (IsDocumentEditable() && IsEditingRenderedHTML())
  3159.     {
  3160.       isEnabled = (GetObjectForProperties() != null ||
  3161.                    GetCurrentEditor().getSelectedElement("href") != null);
  3162.     }
  3163.     return isEnabled;
  3164.   },
  3165.  
  3166.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3167.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3168.  
  3169.   doCommand: function(aCommand)
  3170.   {
  3171.     // Launch Object properties for appropriate selected element 
  3172.     var element = GetObjectForProperties();
  3173.     if (element)
  3174.     {
  3175.       var name = element.nodeName.toLowerCase();
  3176.       switch (name)
  3177.       {
  3178.         case 'img':
  3179.           goDoCommand("cmd_image");
  3180.           break;
  3181.         case 'hr':
  3182.           goDoCommand("cmd_hline");
  3183.           break;
  3184.         case 'form':
  3185.           goDoCommand("cmd_form");
  3186.           break;
  3187.         case 'input':
  3188.           var type = element.getAttribute("type");
  3189.           if (type && type.toLowerCase() == "image")
  3190.             goDoCommand("cmd_inputimage");
  3191.           else
  3192.             goDoCommand("cmd_inputtag");
  3193.           break;
  3194.         case 'textarea':
  3195.           goDoCommand("cmd_textarea");
  3196.           break;
  3197.         case 'select':
  3198.           goDoCommand("cmd_select");
  3199.           break;
  3200.         case 'button':
  3201.           goDoCommand("cmd_button");
  3202.           break;
  3203.         case 'label':
  3204.           goDoCommand("cmd_label");
  3205.           break;
  3206.         case 'fieldset':
  3207.           goDoCommand("cmd_fieldset");
  3208.           break;
  3209.         case 'table':
  3210.           EditorInsertOrEditTable(false);
  3211.           break;
  3212.         case 'td':
  3213.         case 'th':
  3214.           EditorTableCellProperties();
  3215.           break;
  3216.         case 'ol':
  3217.         case 'ul':
  3218.         case 'dl':
  3219.         case 'li':
  3220.           goDoCommand("cmd_listProperties");
  3221.           break;
  3222.         case 'a':
  3223.           if (element.name)
  3224.           {
  3225.             goDoCommand("cmd_anchor");
  3226.           }
  3227.           else if(element.href)
  3228.           {
  3229.             goDoCommand("cmd_link");
  3230.           }
  3231.           break;
  3232.         default:
  3233.           doAdvancedProperties(element);
  3234.           break;
  3235.       }
  3236.     } else {
  3237.       // We get a partially-selected link if asked for specifically
  3238.       try {
  3239.         element = GetCurrentEditor().getSelectedElement("href");
  3240.       } catch (e) {}
  3241.       if (element)
  3242.         goDoCommand("cmd_link");
  3243.     }
  3244.     window.content.focus();
  3245.   }
  3246. };
  3247.  
  3248. //-----------------------------------------------------------------------------------
  3249. var nsGluesCommand =
  3250. {
  3251.   isCommandEnabled: function(aCommand, dummy)
  3252.   {
  3253.     var e = GetCurrentEditor();
  3254.     if (e)
  3255.     {
  3256.       e instanceof Components.interfaces.nsIHTMLAbsPosEditor;
  3257.       var elt = e.positionedElement;
  3258.       if (elt)
  3259.         return true;
  3260.     }
  3261.     return false;
  3262.   },
  3263.  
  3264.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3265.   doCommandParams: function(aCommand, aParams, aRefCon)
  3266.   {
  3267.     var glueDirection = aParams.getCStringValue("state_attribute");
  3268.     var e = GetCurrentEditor();
  3269.     e instanceof Components.interfaces.nsIHTMLAbsPosEditor;
  3270.     var elt = e.positionedElement;
  3271.  
  3272.     // find the enclosing positioned block or the root
  3273.     var enclosing = elt.parentNode;
  3274.     var doc = elt.ownerDocument;
  3275.     while (enclosing != doc.documentElement ||
  3276.            doc.defaultView.getComputedStyle(enclosing, "").getPropertyValue("position") != "static")
  3277.       enclosing = enclosing.parentNode;
  3278.  
  3279.     var newStyle
  3280.     switch (glueDirection)
  3281.     {
  3282.       case "left-pixels":
  3283.         var l = doc.defaultView.getComputedStyle(elt, "").getPropertyCSSValue("left");
  3284.         if (l.cssValueType == CSSValue.CSS_PRIMITIVE_VALUE)
  3285.         {
  3286.           var lpx = l.getFloatValue(CSSPrimitiveValue.CSS_PX);
  3287.           newStyle = "left: " + lpx + "px; right: auto;";
  3288.         }
  3289.         break;
  3290.       case "right-pixels":
  3291.         var r = doc.defaultView.getComputedStyle(elt, "").getPropertyCSSValue("right");
  3292.         if (r.cssValueType == CSSValue.CSS_PRIMITIVE_VALUE)
  3293.         {
  3294.           var rpx = r.getFloatValue(CSSPrimitiveValue.CSS_PX);
  3295.           newStyle = "right: " + rpx + "px; left: auto;";
  3296.         }
  3297.         break;
  3298.       case "center-pixels":
  3299.         var r = doc.defaultView.getComputedStyle(elt, "").getPropertyCSSValue("left");
  3300.         if (r.cssValueType == CSSValue.CSS_PRIMITIVE_VALUE)
  3301.         {
  3302.           var rpx = r.getFloatValue(CSSPrimitiveValue.CSS_PX);
  3303.           var rpc = (rpx / enclosing.realWidth) * 100;
  3304.           newStyle = "left: " + rpc + "%; right: auto;";
  3305.         }
  3306.         break;
  3307.  
  3308.       case "top-pixels":
  3309.         var t = doc.defaultView.getComputedStyle(elt, "").getPropertyCSSValue("top");
  3310.         if (t.cssValueType == CSSValue.CSS_PRIMITIVE_VALUE)
  3311.         {
  3312.           var tpx = t.getFloatValue(CSSPrimitiveValue.CSS_PX);
  3313.           newStyle = "top: " + tpx + "px; bottom: auto;";
  3314.         }
  3315.         break;
  3316.       case "bottom-pixels":
  3317.         var b = doc.defaultView.getComputedStyle(elt, "").getPropertyCSSValue("bottom");
  3318.         if (b.cssValueType == CSSValue.CSS_PRIMITIVE_VALUE)
  3319.         {
  3320.           var bpx = b.getFloatValue(CSSPrimitiveValue.CSS_PX);
  3321.           newStyle = "bottom: " + bpx + "px; top: auto;";
  3322.         }
  3323.         break;
  3324.       case "middle-pixels":
  3325.         var m = doc.defaultView.getComputedStyle(elt, "").getPropertyCSSValue("top");
  3326.         if (m.cssValueType == CSSValue.CSS_PRIMITIVE_VALUE)
  3327.         {
  3328.           var mpx = m.getFloatValue(CSSPrimitiveValue.CSS_PX);
  3329.           var mpc = (mpx / enclosing.realHeight) * 100;
  3330.           newStyle = "top: " + mpc + "%; bottom: auto;";
  3331.         }
  3332.         break;
  3333.  
  3334.       default:
  3335.         break;
  3336.     }
  3337.     if (newStyle)
  3338.     {
  3339.       var style = elt.getAttribute("style");
  3340.       e.setAttribute(elt, "style", style + newStyle);
  3341.     }
  3342.   },
  3343.   // This is now deprecated in favor of "doCommandParams"
  3344.   doCommand: function(aCommand) {}
  3345. };
  3346.  
  3347. //-----------------------------------------------------------------------------------
  3348. var nsSetSmiley =
  3349. {
  3350.   isCommandEnabled: function(aCommand, dummy)
  3351.   {
  3352.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3353.   },
  3354.  
  3355.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3356.   doCommandParams: function(aCommand, aParams, aRefCon)
  3357.   {
  3358.     var smileyCode = aParams.getCStringValue("state_attribute");
  3359.     
  3360.     var strSml;
  3361.     switch(smileyCode)
  3362.     {
  3363.         case ":-)": strSml="s1"; 
  3364.         break;
  3365.         case ":-(": strSml="s2";
  3366.         break;
  3367.         case ";-)": strSml="s3";
  3368.         break;
  3369.         case ":-P": strSml="s4";
  3370.         break;
  3371.         case ":-D": strSml="s5";
  3372.         break;
  3373.         case ":-[": strSml="s6";
  3374.         break;
  3375.         case ":-\\": 
  3376.         case ":\\": strSml="s7";
  3377.         break;
  3378.         case "=-O": strSml="s8";
  3379.         break;
  3380.         case ":-*": strSml="s9";
  3381.         break;
  3382.         case ">:o":
  3383.         case ">:-o": strSml="s10";
  3384.         break;
  3385.         case "8-)": strSml="s11";
  3386.         break;
  3387.         case ":-$": strSml="s12";
  3388.         break;
  3389.         case ":-!": strSml="s13";
  3390.         break;
  3391.         case "O:-)": strSml="s14";
  3392.         break;
  3393.         case ":'(": strSml="s15";
  3394.         break;
  3395.         case ":-X": strSml="s16";
  3396.         break;
  3397.         default:    strSml="";
  3398.         break;
  3399.     }
  3400.  
  3401.     try 
  3402.     {
  3403.       var editor = GetCurrentEditor();
  3404.       var selection = editor.selection;
  3405.       var extElement = editor.createElementWithDefaults("span");
  3406.       extElement.setAttribute("class", "moz-smiley-" + strSml);
  3407.  
  3408.       var intElement = editor.createElementWithDefaults("span");
  3409.       if (!intElement)
  3410.         return;
  3411.  
  3412.       //just for mailnews, because of the way it removes HTML
  3413.       var smileButMenu = document.getElementById('smileButtonMenu');      
  3414.       if (smileButMenu.getAttribute("padwithspace"))
  3415.          smileyCode = " " + smileyCode + " ";
  3416.  
  3417.       var txtElement =  editor.document.createTextNode(smileyCode);
  3418.       if (!txtElement)
  3419.         return;
  3420.  
  3421.       intElement.appendChild (txtElement);
  3422.       extElement.appendChild (intElement);
  3423.  
  3424.  
  3425.       editor.insertElementAtSelection(extElement,true);
  3426.       window.content.focus();        
  3427.  
  3428.     } 
  3429.     catch (e) 
  3430.     {
  3431.         dump("Exception occured in smiley InsertElementAtSelection\n");
  3432.     }
  3433.   },
  3434.   // This is now deprecated in favor of "doCommandParams"
  3435.   doCommand: function(aCommand) {}
  3436. };
  3437.  
  3438.  
  3439. function doAdvancedProperties(element)
  3440. {
  3441.   if (element)
  3442.   {
  3443.     window.openDialog("chrome://editor/content/EdAdvancedEdit.xul", "_blank", "chrome,close,titlebar,modal,resizable=yes", "", element);
  3444.     window.content.focus();
  3445.   }
  3446. }
  3447.  
  3448. var nsAdvancedPropertiesCommand =
  3449. {
  3450.   isCommandEnabled: function(aCommand, dummy)
  3451.   {
  3452.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3453.   },
  3454.  
  3455.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3456.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3457.  
  3458.   doCommand: function(aCommand)
  3459.   {
  3460.     // Launch AdvancedEdit dialog for the selected element
  3461.     try {
  3462.       var element = GetCurrentEditor().getSelectedElement("");
  3463.       doAdvancedProperties(element);
  3464.     } catch (e) {}
  3465.   }
  3466. };
  3467.  
  3468. //-----------------------------------------------------------------------------------
  3469. var nsColorPropertiesCommand =
  3470. {
  3471.   isCommandEnabled: function(aCommand, dummy)
  3472.   {
  3473.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3474.   },
  3475.  
  3476.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3477.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3478.  
  3479.   doCommand: function(aCommand)
  3480.   {
  3481.     window.openDialog("chrome://editor/content/EdColorProps.xul","_blank", "chrome,close,titlebar,modal", ""); 
  3482.     UpdateDefaultColors(); 
  3483.     window.content.focus();
  3484.   }
  3485. };
  3486.  
  3487. //-----------------------------------------------------------------------------------
  3488. var nsRemoveNamedAnchorsCommand =
  3489. {
  3490.   isCommandEnabled: function(aCommand, dummy)
  3491.   {
  3492.     // We could see if there's any link in selection, but it doesn't seem worth the work!
  3493.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3494.   },
  3495.  
  3496.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3497.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3498.  
  3499.   doCommand: function(aCommand)
  3500.   {
  3501.     EditorRemoveTextProperty("name", "");
  3502.     window.content.focus();
  3503.   }
  3504. };
  3505.  
  3506.  
  3507. //-----------------------------------------------------------------------------------
  3508. var nsEditLinkCommand =
  3509. {
  3510.   isCommandEnabled: function(aCommand, dummy)
  3511.   {
  3512.     // Not really used -- this command is only in context menu, and we do enabling there
  3513.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3514.   },
  3515.  
  3516.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3517.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3518.  
  3519.   doCommand: function(aCommand)
  3520.   {
  3521.     try {
  3522.       var element = GetCurrentEditor().getSelectedElement("href");
  3523.       if (element)
  3524.         editPage(element.href, window, false, true);
  3525.     } catch (e) {}
  3526.     window.content.focus();
  3527.   }
  3528. };
  3529.  
  3530.  
  3531. //-----------------------------------------------------------------------------------
  3532. var nsNormalModeCommand =
  3533. {
  3534.   isCommandEnabled: function(aCommand, dummy)
  3535.   {
  3536.     return IsHTMLEditor() && IsDocumentEditable();
  3537.   },
  3538.  
  3539.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3540.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3541.  
  3542.   doCommand: function(aCommand)
  3543.   {
  3544.     SetEditMode(kDisplayModeNormal);
  3545.   }
  3546. };
  3547.  
  3548. var nsAllTagsModeCommand =
  3549. {
  3550.   isCommandEnabled: function(aCommand, dummy)
  3551.   {
  3552.     return (IsDocumentEditable() && IsHTMLEditor());
  3553.   },
  3554.  
  3555.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3556.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3557.  
  3558.   doCommand: function(aCommand)
  3559.   {
  3560.     SetEditMode(kDisplayModeAllTags);
  3561.   }
  3562. };
  3563.  
  3564. var nsHTMLSourceModeCommand =
  3565. {
  3566.   isCommandEnabled: function(aCommand, dummy)
  3567.   {
  3568.     return (IsDocumentEditable() && IsHTMLEditor());
  3569.   },
  3570.  
  3571.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3572.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3573.  
  3574.   doCommand: function(aCommand)
  3575.   {
  3576.     SetEditMode(kDisplayModeSource);
  3577.   }
  3578. };
  3579.  
  3580. var nsPreviewModeCommand =
  3581. {
  3582.   isCommandEnabled: function(aCommand, dummy)
  3583.   {
  3584.     return (IsDocumentEditable() && IsHTMLEditor());
  3585.   },
  3586.  
  3587.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3588.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3589.  
  3590.   doCommand: function(aCommand)
  3591.   {
  3592.     SetEditMode(kDisplayModePreview);
  3593.   }
  3594. };
  3595.  
  3596. //-----------------------------------------------------------------------------------
  3597. var nsInsertOrEditTableCommand =
  3598. {
  3599.   isCommandEnabled: function(aCommand, dummy)
  3600.   {
  3601.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  3602.   },
  3603.  
  3604.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3605.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3606.  
  3607.   doCommand: function(aCommand)
  3608.   {
  3609.     if (IsInTableCell())
  3610.       EditorTableCellProperties();
  3611.     else
  3612.       EditorInsertOrEditTable(true);
  3613.   }
  3614. };
  3615.  
  3616. //-----------------------------------------------------------------------------------
  3617. var nsEditTableCommand =
  3618. {
  3619.   isCommandEnabled: function(aCommand, dummy)
  3620.   {
  3621.     return IsInTable();
  3622.   },
  3623.  
  3624.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3625.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3626.  
  3627.   doCommand: function(aCommand)
  3628.   {
  3629.     EditorInsertOrEditTable(false);
  3630.   }
  3631. };
  3632.  
  3633. //-----------------------------------------------------------------------------------
  3634. var nsSelectTableCommand =
  3635. {
  3636.   isCommandEnabled: function(aCommand, dummy)
  3637.   {
  3638.     return IsInTable();
  3639.   },
  3640.  
  3641.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3642.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3643.  
  3644.   doCommand: function(aCommand)
  3645.   {
  3646.     try {
  3647.       GetCurrentTableEditor().selectTable();
  3648.     } catch(e) {}
  3649.     window.content.focus();
  3650.   }
  3651. };
  3652.  
  3653. var nsSelectTableRowCommand =
  3654. {
  3655.   isCommandEnabled: function(aCommand, dummy)
  3656.   {
  3657.     return IsInTableCell();
  3658.   },
  3659.  
  3660.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3661.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3662.  
  3663.   doCommand: function(aCommand)
  3664.   {
  3665.     try {
  3666.       GetCurrentTableEditor().selectTableRow();
  3667.     } catch(e) {}
  3668.     window.content.focus();
  3669.   }
  3670. };
  3671.  
  3672. var nsSelectTableColumnCommand =
  3673. {
  3674.   isCommandEnabled: function(aCommand, dummy)
  3675.   {
  3676.     return IsInTableCell();
  3677.   },
  3678.  
  3679.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3680.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3681.  
  3682.   doCommand: function(aCommand)
  3683.   {
  3684.     try {
  3685.       GetCurrentTableEditor().selectTableColumn();
  3686.     } catch(e) {}
  3687.     window.content.focus();
  3688.   }
  3689. };
  3690.  
  3691. var nsSelectTableCellCommand =
  3692. {
  3693.   isCommandEnabled: function(aCommand, dummy)
  3694.   {
  3695.     return IsInTableCell();
  3696.   },
  3697.  
  3698.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3699.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3700.  
  3701.   doCommand: function(aCommand)
  3702.   {
  3703.     try {
  3704.       GetCurrentTableEditor().selectTableCell();
  3705.     } catch(e) {}
  3706.     window.content.focus();
  3707.   }
  3708. };
  3709.  
  3710. var nsSelectAllTableCellsCommand =
  3711. {
  3712.   isCommandEnabled: function(aCommand, dummy)
  3713.   {
  3714.     return IsInTable();
  3715.   },
  3716.  
  3717.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3718.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3719.  
  3720.   doCommand: function(aCommand)
  3721.   {
  3722.     try {
  3723.       GetCurrentTableEditor().selectAllTableCells();
  3724.     } catch(e) {}
  3725.     window.content.focus();
  3726.   }
  3727. };
  3728.  
  3729. //-----------------------------------------------------------------------------------
  3730. var nsInsertTableCommand =
  3731. {
  3732.   isCommandEnabled: function(aCommand, dummy)
  3733.   {
  3734.     return IsDocumentEditable() && IsEditingRenderedHTML();
  3735.   },
  3736.  
  3737.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3738.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3739.  
  3740.   doCommand: function(aCommand)
  3741.   {
  3742.     EditorInsertTable();
  3743.   }
  3744. };
  3745.  
  3746. var nsInsertTableRowAboveCommand =
  3747. {
  3748.   isCommandEnabled: function(aCommand, dummy)
  3749.   {
  3750.     return IsInTableCell();
  3751.   },
  3752.  
  3753.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3754.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3755.  
  3756.   doCommand: function(aCommand)
  3757.   {
  3758.     try {
  3759.       GetCurrentTableEditor().insertTableRow(1, false);
  3760.     } catch(e) {}
  3761.     window.content.focus();
  3762.   }
  3763. };
  3764.  
  3765. var nsInsertTableRowBelowCommand =
  3766. {
  3767.   isCommandEnabled: function(aCommand, dummy)
  3768.   {
  3769.     return IsInTableCell();
  3770.   },
  3771.  
  3772.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3773.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3774.  
  3775.   doCommand: function(aCommand)
  3776.   {
  3777.     try {
  3778.       GetCurrentTableEditor().insertTableRow(1, true);
  3779.     } catch(e) {}
  3780.     window.content.focus();
  3781.   }
  3782. };
  3783.  
  3784. var nsInsertTableColumnBeforeCommand =
  3785. {
  3786.   isCommandEnabled: function(aCommand, dummy)
  3787.   {
  3788.     return IsInTableCell();
  3789.   },
  3790.  
  3791.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3792.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3793.  
  3794.   doCommand: function(aCommand)
  3795.   {
  3796.     try {
  3797.       GetCurrentTableEditor().insertTableColumn(1, false);
  3798.     } catch(e) {}
  3799.     window.content.focus();
  3800.   }
  3801. };
  3802.  
  3803. var nsInsertTableColumnAfterCommand =
  3804. {
  3805.   isCommandEnabled: function(aCommand, dummy)
  3806.   {
  3807.     return IsInTableCell();
  3808.   },
  3809.  
  3810.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3811.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3812.  
  3813.   doCommand: function(aCommand)
  3814.   {
  3815.     try {
  3816.       GetCurrentTableEditor().insertTableColumn(1, true);
  3817.     } catch(e) {}
  3818.     window.content.focus();
  3819.   }
  3820. };
  3821.  
  3822. var nsInsertTableCellBeforeCommand =
  3823. {
  3824.   isCommandEnabled: function(aCommand, dummy)
  3825.   {
  3826.     return IsInTableCell();
  3827.   },
  3828.  
  3829.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3830.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3831.  
  3832.   doCommand: function(aCommand)
  3833.   {
  3834.     try {
  3835.       GetCurrentTableEditor().insertTableCell(1, false);
  3836.     } catch(e) {}
  3837.     window.content.focus();
  3838.   }
  3839. };
  3840.  
  3841. var nsInsertTableCellAfterCommand =
  3842. {
  3843.   isCommandEnabled: function(aCommand, dummy)
  3844.   {
  3845.     return IsInTableCell();
  3846.   },
  3847.  
  3848.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3849.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3850.  
  3851.   doCommand: function(aCommand)
  3852.   {
  3853.     try {
  3854.       GetCurrentTableEditor().insertTableCell(1, true);
  3855.     } catch(e) {}
  3856.     window.content.focus();
  3857.   }
  3858. };
  3859.  
  3860. //-----------------------------------------------------------------------------------
  3861. var nsDeleteTableCommand =
  3862. {
  3863.   isCommandEnabled: function(aCommand, dummy)
  3864.   {
  3865.     return IsInTable();
  3866.   },
  3867.  
  3868.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3869.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3870.  
  3871.   doCommand: function(aCommand)
  3872.   {
  3873.     try {
  3874.       GetCurrentTableEditor().deleteTable();
  3875.     } catch(e) {}
  3876.     window.content.focus();
  3877.   }
  3878. };
  3879.  
  3880. var nsDeleteTableRowCommand =
  3881. {
  3882.   isCommandEnabled: function(aCommand, dummy)
  3883.   {
  3884.     return IsInTableCell();
  3885.   },
  3886.  
  3887.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3888.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3889.  
  3890.   doCommand: function(aCommand)
  3891.   {
  3892.     var rows = GetNumberOfContiguousSelectedRows();
  3893.     // Delete at least one row
  3894.     if (rows == 0)
  3895.       rows = 1;
  3896.  
  3897.     try {
  3898.       var editor = GetCurrentTableEditor();
  3899.       editor.beginTransaction();
  3900.  
  3901.       // Loop to delete all blocks of contiguous, selected rows
  3902.       while (rows)
  3903.       {
  3904.         editor.deleteTableRow(rows);
  3905.         rows = GetNumberOfContiguousSelectedRows();
  3906.       }
  3907.     } finally { editor.endTransaction(); }
  3908.     window.content.focus();
  3909.   }
  3910. };
  3911.  
  3912. var nsDeleteTableColumnCommand =
  3913. {
  3914.   isCommandEnabled: function(aCommand, dummy)
  3915.   {
  3916.     return IsInTableCell();
  3917.   },
  3918.  
  3919.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3920.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3921.  
  3922.   doCommand: function(aCommand)
  3923.   {
  3924.     var columns = GetNumberOfContiguousSelectedColumns();
  3925.     // Delete at least one column
  3926.     if (columns == 0)
  3927.       columns = 1;
  3928.  
  3929.     try {
  3930.       var editor = GetCurrentTableEditor();
  3931.       editor.beginTransaction();
  3932.  
  3933.       // Loop to delete all blocks of contiguous, selected columns
  3934.       while (columns)
  3935.       {
  3936.         editor.deleteTableColumn(columns);
  3937.         columns = GetNumberOfContiguousSelectedColumns();
  3938.       }
  3939.     } finally { editor.endTransaction(); }
  3940.     window.content.focus();
  3941.   }
  3942. };
  3943.  
  3944. var nsDeleteTableCellCommand =
  3945. {
  3946.   isCommandEnabled: function(aCommand, dummy)
  3947.   {
  3948.     return IsInTableCell();
  3949.   },
  3950.  
  3951.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3952.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3953.  
  3954.   doCommand: function(aCommand)
  3955.   {
  3956.     try {
  3957.       GetCurrentTableEditor().deleteTableCell(1);   
  3958.     } catch (e) {}
  3959.     window.content.focus();
  3960.   }
  3961. };
  3962.  
  3963. var nsDeleteTableCellContentsCommand =
  3964. {
  3965.   isCommandEnabled: function(aCommand, dummy)
  3966.   {
  3967.     return IsInTableCell();
  3968.   },
  3969.  
  3970.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3971.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3972.  
  3973.   doCommand: function(aCommand)
  3974.   {
  3975.     try {
  3976.       GetCurrentTableEditor().deleteTableCellContents();   
  3977.     } catch (e) {}
  3978.     window.content.focus();
  3979.   }
  3980. };
  3981.  
  3982.  
  3983. //-----------------------------------------------------------------------------------
  3984. var nsNormalizeTableCommand =
  3985. {
  3986.   isCommandEnabled: function(aCommand, dummy)
  3987.   {
  3988.     return IsInTable();
  3989.   },
  3990.  
  3991.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  3992.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  3993.  
  3994.   doCommand: function(aCommand)
  3995.   {
  3996.     // Use nsnull to let editor find table enclosing current selection
  3997.     try {
  3998.       GetCurrentTableEditor().normalizeTable(null);   
  3999.     } catch (e) {}
  4000.     window.content.focus();
  4001.   }
  4002. };
  4003.  
  4004. //-----------------------------------------------------------------------------------
  4005. var nsJoinTableCellsCommand =
  4006. {
  4007.   isCommandEnabled: function(aCommand, dummy)
  4008.   {
  4009.     if (IsDocumentEditable() && IsEditingRenderedHTML())
  4010.     {
  4011.       try {
  4012.         var editor = GetCurrentTableEditor();
  4013.         var tagNameObj = { value: "" };
  4014.         var countObj = { value: 0 };
  4015.         var cell = editor.getSelectedOrParentTableElement(tagNameObj, countObj);
  4016.  
  4017.         // We need a cell and either > 1 selected cell or a cell to the right
  4018.         //  (this cell may originate in a row spanned from above current row)
  4019.         // Note that editor returns "td" for "th" also.
  4020.         // (this is a pain! Editor and gecko use lowercase tagNames, JS uses uppercase!)
  4021.         if( cell && (tagNameObj.value == "td"))
  4022.         {
  4023.           // Selected cells
  4024.           if (countObj.value > 1) return true;
  4025.  
  4026.           var colSpan = cell.getAttribute("colspan");
  4027.  
  4028.           // getAttribute returns string, we need number
  4029.           // no attribute means colspan = 1
  4030.           if (!colSpan)
  4031.             colSpan = Number(1);
  4032.           else
  4033.             colSpan = Number(colSpan);
  4034.  
  4035.           var rowObj = { value: 0 };
  4036.           var colObj = { value: 0 };
  4037.           editor.getCellIndexes(cell, rowObj, colObj);
  4038.  
  4039.           // Test if cell exists to the right of current cell
  4040.           // (cells with 0 span should never have cells to the right
  4041.           //  if there is, user can select the 2 cells to join them)
  4042.           return (colSpan && editor.getCellAt(null, rowObj.value,
  4043.                                               colObj.value + colSpan));
  4044.         }
  4045.       } catch (e) {}
  4046.     }
  4047.     return false;
  4048.   },
  4049.  
  4050.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4051.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4052.  
  4053.   doCommand: function(aCommand)
  4054.   {
  4055.     // Param: Don't merge non-contiguous cells
  4056.     try {
  4057.       GetCurrentTableEditor().joinTableCells(false);
  4058.     } catch (e) {}
  4059.     window.content.focus();
  4060.   }
  4061. };
  4062.  
  4063. //-----------------------------------------------------------------------------------
  4064. var nsSplitTableCellCommand =
  4065. {
  4066.   isCommandEnabled: function(aCommand, dummy)
  4067.   {
  4068.     if (IsDocumentEditable() && IsEditingRenderedHTML())
  4069.     {
  4070.       var tagNameObj = { value: "" };
  4071.       var countObj = { value: 0 };
  4072.       var cell;
  4073.       try {
  4074.         cell = GetCurrentTableEditor().getSelectedOrParentTableElement(tagNameObj, countObj);
  4075.       } catch (e) {}
  4076.  
  4077.       // We need a cell parent and there's just 1 selected cell 
  4078.       // or selection is entirely inside 1 cell
  4079.       if ( cell && (tagNameObj.value == "td") && 
  4080.            countObj.value <= 1 &&
  4081.            IsSelectionInOneCell() )
  4082.       {
  4083.         var colSpan = cell.getAttribute("colspan");
  4084.         var rowSpan = cell.getAttribute("rowspan");
  4085.         if (!colSpan) colSpan = 1;
  4086.         if (!rowSpan) rowSpan = 1;
  4087.         return (colSpan > 1  || rowSpan > 1 ||
  4088.                 colSpan == 0 || rowSpan == 0);
  4089.       }
  4090.     }
  4091.     return false;
  4092.   },
  4093.  
  4094.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4095.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4096.  
  4097.   doCommand: function(aCommand)
  4098.   {
  4099.     try {
  4100.       GetCurrentTableEditor().splitTableCell();
  4101.     } catch (e) {}
  4102.     window.content.focus();
  4103.   }
  4104. };
  4105.  
  4106. //-----------------------------------------------------------------------------------
  4107. var nsTableOrCellColorCommand =
  4108. {
  4109.   isCommandEnabled: function(aCommand, dummy)
  4110.   {
  4111.     return IsInTable();
  4112.   },
  4113.  
  4114.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4115.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4116.  
  4117.   doCommand: function(aCommand)
  4118.   {
  4119.     EditorSelectColor("TableOrCell");
  4120.   }
  4121. };
  4122.  
  4123. //-----------------------------------------------------------------------------------
  4124. var nsPreferencesCommand =
  4125. {
  4126.   isCommandEnabled: function(aCommand, dummy)
  4127.   {
  4128.     return true;
  4129.   },
  4130.  
  4131.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4132.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4133.  
  4134.   doCommand: function(aCommand)
  4135.   {
  4136.     goPreferences('editor', 'chrome://editor/content/pref-composer.xul','editor');
  4137.     window.content.focus();
  4138.   }
  4139. };
  4140.  
  4141. //-----------------------------------------------------------------------------------
  4142. var nsBlockBordersCommand =
  4143. {
  4144.   isCommandEnabled: function(aCommand, dummy)
  4145.   {
  4146.     return (IsDocumentEditable() && IsEditingRenderedHTML());
  4147.   },
  4148.  
  4149.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4150.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4151.  
  4152.   doCommand: function(aCommand)
  4153.   {
  4154.     window.openDialog("chrome://cascades/content/borderProps.xul","_blank", "chrome,close,modal,titlebar",
  4155.                       "border", "Borders", true);
  4156.     window.content.focus();
  4157.   }
  4158. };
  4159.  
  4160. var nsFinishHTMLSource =
  4161. {
  4162.   isCommandEnabled: function(aCommand, dummy)
  4163.   {
  4164.     return true;
  4165.   },
  4166.  
  4167.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4168.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4169.  
  4170.   doCommand: function(aCommand)
  4171.   {
  4172.     // In editor.js
  4173.     FinishHTMLSource();
  4174.   }
  4175. };
  4176.  
  4177. var nsCancelHTMLSource =
  4178. {
  4179.   isCommandEnabled: function(aCommand, dummy)
  4180.   {
  4181.     return true;
  4182.   },
  4183.  
  4184.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4185.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4186.  
  4187.   doCommand: function(aCommand)
  4188.   {
  4189.     // In editor.js
  4190.     CancelHTMLSource();
  4191.   }
  4192. };
  4193.  
  4194. var nsConvertToTable =
  4195. {
  4196.   isCommandEnabled: function(aCommand, dummy)
  4197.   {
  4198.     if (IsDocumentEditable() && IsEditingRenderedHTML())
  4199.     {
  4200.       var selection;
  4201.       try {
  4202.         selection = GetCurrentEditor().selection;
  4203.       } catch (e) {}
  4204.  
  4205.       if (selection && !selection.isCollapsed)
  4206.       {
  4207.         // Don't allow if table or cell is the selection
  4208.         var element;
  4209.         try {
  4210.           element = GetCurrentEditor().getSelectedElement("");
  4211.         } catch (e) {}
  4212.         if (element)
  4213.         {
  4214.           var name = element.nodeName.toLowerCase();
  4215.           if (name == "td" ||
  4216.               name == "th" ||
  4217.               name == "caption" ||
  4218.               name == "table")
  4219.             return false;
  4220.         }
  4221.  
  4222.         // Selection start and end must be in the same cell
  4223.         //   in same cell or both are NOT in a cell
  4224.         if ( GetParentTableCell(selection.focusNode) !=
  4225.              GetParentTableCell(selection.anchorNode) )
  4226.           return false
  4227.       
  4228.         return true;
  4229.       }
  4230.     }
  4231.     return false;
  4232.   },
  4233.  
  4234.   getCommandStateParams: function(aCommand, aParams, aRefCon) {},
  4235.   doCommandParams: function(aCommand, aParams, aRefCon) {},
  4236.  
  4237.   doCommand: function(aCommand)
  4238.   {
  4239.     if (this.isCommandEnabled())
  4240.     {
  4241.       window.openDialog("chrome://editor/content/EdConvertToTable.xul","_blank", "chrome,close,titlebar,modal")
  4242.     }
  4243.     window.content.focus();
  4244.   }
  4245. };
  4246.  
  4247.